Skip to content

Commit fe67ef2

Browse files
galenlynchstevengj
authored andcommitted
Make Julia exceptions wrapped in Python Exceptions more informative (#608)
* Make Julia exceptions wrapped in Python Exceptions more informative I have added a backtrace to the description of exceptions coming from Julia that are wrapped in Python exceptions. This makes it much easier to find the source of errors in Julia code being called from Python functions. * Allow `pyraise` to include backtrace from Julia Instead of using `catch_backtrace` in `showerror_string`, which may not always be called from a catch block with a current exception set, I have added an optional argument to `pyraise` which can be used to pass backtraces to `showerror_string`. Backtraces are only used if exception is not a PyError exception. By default, no backtrace is shown. All try-catch blocks in PyCall attempt to pass the backtrace for the current exception to `pyraise`. * Add pyraise macro Added a pyraise macro, as written by @stevengj, which catches backtraces and calls the pyraise function. This macro should only be used in a catch block. * Allow ioraise to display Julia backtrace
1 parent 8eb62f2 commit fe67ef2

File tree

5 files changed

+31
-15
lines changed

5 files changed

+31
-15
lines changed

src/callback.jl

+1-1
Original file line numberDiff line numberDiff line change
@@ -39,7 +39,7 @@ function _pyjlwrap_call(f, args_::PyPtr, kw_::PyPtr)
3939

4040
return pyreturn(ret)
4141
catch e
42-
pyraise(e)
42+
@pyraise e
4343
finally
4444
args.o = PyPtr_NULL # don't decref
4545
end

src/exception.jl

+21-5
Original file line numberDiff line numberDiff line change
@@ -123,16 +123,21 @@ function pyexc_initialize()
123123
pyexc[PyIOError] = @pyglobalobjptr :PyExc_IOError
124124
end
125125

126+
_showerror_string(io::IO, e, ::Nothing) = showerror(io, e)
127+
_showerror_string(io::IO, e, bt) = showerror(io, e, bt)
128+
129+
# bt argument defaults to nothing, to delay dispatching on the presence of a
130+
# backtrace until after the try-catch block
126131
"""
127132
showerror_string(e) :: String
128133
129134
Convert output of `showerror` to a `String`. Since this function may
130135
be called via Python C-API, it tries to not throw at all cost.
131136
"""
132-
function showerror_string(e::T) where {T}
137+
function showerror_string(e::T, bt = nothing) where {T}
133138
try
134139
io = IOBuffer()
135-
showerror(io, e)
140+
_showerror_string(io, e, bt)
136141
return String(take!(io))
137142
catch
138143
try
@@ -163,15 +168,26 @@ function showerror_string(e::T) where {T}
163168
end
164169
end
165170

166-
function pyraise(e)
171+
function pyraise(e, bt = nothing)
167172
eT = typeof(e)
168173
pyeT = haskey(pyexc::Dict, eT) ? pyexc[eT] : pyexc[Exception]
169174
ccall((@pysym :PyErr_SetString), Cvoid, (PyPtr, Cstring),
170-
pyeT, string("Julia exception: ", showerror_string(e)))
175+
pyeT, string("Julia exception: ", showerror_string(e, bt)))
171176
end
172177

173-
function pyraise(e::PyError)
178+
# Second argument allows for backtraces passed to `pyraise` to be ignored.
179+
function pyraise(e::PyError, ::Vector = [])
174180
ccall((@pysym :PyErr_Restore), Cvoid, (PyPtr, PyPtr, PyPtr),
175181
e.T, e.val, e.traceback)
176182
e.T.o = e.val.o = e.traceback.o = C_NULL # refs were stolen
177183
end
184+
185+
"""
186+
@pyraise e
187+
188+
Throw the exception `e` to Python, attaching a backtrace. This macro should only be
189+
used in a `catch` block so that `catch_backtrace()` is valid.
190+
"""
191+
macro pyraise(e)
192+
:(pyraise($(esc(e)), catch_backtrace()))
193+
end

src/io.jl

+4-4
Original file line numberDiff line numberDiff line change
@@ -8,21 +8,21 @@
88
# IOBase methods:
99

1010
# IO objects should raise IOError for unsupported operations or failed IO
11-
function ioraise(e)
11+
function ioraise(e, bt = nothing)
1212
if isa(e, MethodError) || isa(e, ErrorException)
1313
ccall((@pysym :PyErr_SetString), Cvoid, (PyPtr, Cstring),
1414
(pyexc::Dict)[PyIOError],
15-
string("Julia exception: ", e))
15+
showerror_string(e, bt))
1616
else
17-
pyraise(e)
17+
pyraise(e, bt)
1818
end
1919
end
2020

2121
macro with_ioraise(expr)
2222
:(try
2323
$(esc(expr))
2424
catch e
25-
ioraise(e)
25+
ioraise(e, catch_backtrace())
2626
end)
2727
end
2828

src/pyiterator.jl

+3-3
Original file line numberDiff line numberDiff line change
@@ -134,7 +134,7 @@ const jlWrapIteratorType = PyTypeObject()
134134
return pyreturn(item)
135135
end
136136
catch e
137-
pyraise(e)
137+
@pyraise e
138138
end
139139
return PyPtr_NULL
140140
end
@@ -149,7 +149,7 @@ else
149149
return pyreturn(item)
150150
end
151151
catch e
152-
pyraise(e)
152+
@pyraise e
153153
end
154154
return PyPtr_NULL
155155
end
@@ -162,7 +162,7 @@ function pyjlwrap_getiter(self_::PyPtr)
162162
self = unsafe_pyjlwrap_to_objref(self_)
163163
return pystealref!(jlwrap_iterator(self))
164164
catch e
165-
pyraise(e)
165+
@pyraise e
166166
end
167167
return PyPtr_NULL
168168
end

src/pytype.jl

+2-2
Original file line numberDiff line numberDiff line change
@@ -348,7 +348,7 @@ function pyjlwrap_repr(o::PyPtr)
348348
return pyreturn(o != C_NULL ? string("<PyCall.jlwrap ",unsafe_pyjlwrap_to_objref(o),">")
349349
: "<PyCall.jlwrap NULL>")
350350
catch e
351-
pyraise(e)
351+
@pyraise e
352352
return PyPtr_NULL
353353
end
354354
end
@@ -395,7 +395,7 @@ function pyjlwrap_getattr(self_::PyPtr, attr__::PyPtr)
395395
end
396396
end
397397
catch e
398-
pyraise(e)
398+
@pyraise e
399399
finally
400400
attr_.o = PyPtr_NULL # don't decref
401401
end

0 commit comments

Comments
 (0)