mirror of
https://github.com/ruby/ruby.git
synced 2022-11-09 12:17:21 -05:00
Fix for circular causes
* eval_error.c (show_cause): get rid of infinite recursion on circular causes. [Bug #15447] git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@66493 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
This commit is contained in:
parent
65dced4320
commit
1849288722
2 changed files with 33 additions and 6 deletions
26
eval_error.c
26
eval_error.c
|
@ -222,23 +222,36 @@ print_backtrace(const VALUE eclass, const VALUE errat, const VALUE str, int reve
|
||||||
|
|
||||||
VALUE rb_get_message(VALUE exc);
|
VALUE rb_get_message(VALUE exc);
|
||||||
|
|
||||||
|
static int
|
||||||
|
shown_cause_p(VALUE cause, VALUE *shown_causes)
|
||||||
|
{
|
||||||
|
VALUE shown = *shown_causes;
|
||||||
|
if (!shown) {
|
||||||
|
*shown_causes = shown = rb_obj_hide(rb_ident_hash_new());
|
||||||
|
}
|
||||||
|
if (rb_hash_has_key(shown, cause)) return TRUE;
|
||||||
|
rb_hash_aset(shown, cause, Qtrue);
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
show_cause(VALUE errinfo, VALUE str, VALUE highlight, VALUE reverse)
|
show_cause(VALUE errinfo, VALUE str, VALUE highlight, VALUE reverse, VALUE *shown_causes)
|
||||||
{
|
{
|
||||||
VALUE cause = rb_attr_get(errinfo, id_cause);
|
VALUE cause = rb_attr_get(errinfo, id_cause);
|
||||||
if (!NIL_P(cause) && rb_obj_is_kind_of(cause, rb_eException)) {
|
if (!NIL_P(cause) && rb_obj_is_kind_of(cause, rb_eException) &&
|
||||||
|
!shown_cause_p(cause, shown_causes)) {
|
||||||
volatile VALUE eclass = CLASS_OF(cause);
|
volatile VALUE eclass = CLASS_OF(cause);
|
||||||
VALUE errat = rb_get_backtrace(cause);
|
VALUE errat = rb_get_backtrace(cause);
|
||||||
VALUE emesg = rb_get_message(cause);
|
VALUE emesg = rb_get_message(cause);
|
||||||
if (reverse) {
|
if (reverse) {
|
||||||
show_cause(cause, str, highlight, reverse);
|
show_cause(cause, str, highlight, reverse, shown_causes);
|
||||||
print_backtrace(eclass, errat, str, TRUE);
|
print_backtrace(eclass, errat, str, TRUE);
|
||||||
print_errinfo(eclass, errat, emesg, str, highlight!=0);
|
print_errinfo(eclass, errat, emesg, str, highlight!=0);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
print_errinfo(eclass, errat, emesg, str, highlight!=0);
|
print_errinfo(eclass, errat, emesg, str, highlight!=0);
|
||||||
print_backtrace(eclass, errat, str, FALSE);
|
print_backtrace(eclass, errat, str, FALSE);
|
||||||
show_cause(cause, str, highlight, reverse);
|
show_cause(cause, str, highlight, reverse, shown_causes);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -247,6 +260,7 @@ void
|
||||||
rb_error_write(VALUE errinfo, VALUE emesg, VALUE errat, VALUE str, VALUE highlight, VALUE reverse)
|
rb_error_write(VALUE errinfo, VALUE emesg, VALUE errat, VALUE str, VALUE highlight, VALUE reverse)
|
||||||
{
|
{
|
||||||
volatile VALUE eclass;
|
volatile VALUE eclass;
|
||||||
|
VALUE shown_causes = 0;
|
||||||
|
|
||||||
if (NIL_P(errinfo))
|
if (NIL_P(errinfo))
|
||||||
return;
|
return;
|
||||||
|
@ -277,14 +291,14 @@ rb_error_write(VALUE errinfo, VALUE emesg, VALUE errat, VALUE str, VALUE highlig
|
||||||
len = p - (msg = buff);
|
len = p - (msg = buff);
|
||||||
}
|
}
|
||||||
write_warn2(str, msg, len);
|
write_warn2(str, msg, len);
|
||||||
show_cause(errinfo, str, highlight, reverse);
|
show_cause(errinfo, str, highlight, reverse, &shown_causes);
|
||||||
print_backtrace(eclass, errat, str, TRUE);
|
print_backtrace(eclass, errat, str, TRUE);
|
||||||
print_errinfo(eclass, errat, emesg, str, highlight!=0);
|
print_errinfo(eclass, errat, emesg, str, highlight!=0);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
print_errinfo(eclass, errat, emesg, str, highlight!=0);
|
print_errinfo(eclass, errat, emesg, str, highlight!=0);
|
||||||
print_backtrace(eclass, errat, str, FALSE);
|
print_backtrace(eclass, errat, str, FALSE);
|
||||||
show_cause(errinfo, str, highlight, reverse);
|
show_cause(errinfo, str, highlight, reverse, &shown_causes);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1353,6 +1353,19 @@ $stderr = $stdout; raise "\x82\xa0"') do |outs, errs, status|
|
||||||
assert_in_out_err([], code, [], /foo/, success: false, timeout: 2)
|
assert_in_out_err([], code, [], /foo/, success: false, timeout: 2)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def test_circular_cause_handle
|
||||||
|
errs = [/.*error 1.*\n/, /.*error 2.*\n/, /.*error 1.*/m]
|
||||||
|
assert_in_out_err([], "#{<<~"begin;"}\n#{<<~'end;'}", [], errs, success: false, timeout: 2)
|
||||||
|
begin;
|
||||||
|
begin
|
||||||
|
raise "error 1"
|
||||||
|
rescue => e1
|
||||||
|
raise "error 2" rescue e2 = $!
|
||||||
|
raise e1, cause: e2
|
||||||
|
end
|
||||||
|
end;
|
||||||
|
end
|
||||||
|
|
||||||
def test_super_in_method_missing
|
def test_super_in_method_missing
|
||||||
assert_separately([], "#{<<~"begin;"}\n#{<<~'end;'}")
|
assert_separately([], "#{<<~"begin;"}\n#{<<~'end;'}")
|
||||||
begin;
|
begin;
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue