Show errors, even in production.
This commit is contained in:
parent
60708b5bb7
commit
af1640abec
34
main.py
34
main.py
|
@ -2,19 +2,51 @@
|
||||||
"""Find photos on flickr for Wikipedia articls and contact the photographer."""
|
"""Find photos on flickr for Wikipedia articls and contact the photographer."""
|
||||||
|
|
||||||
import collections
|
import collections
|
||||||
|
import inspect
|
||||||
import json
|
import json
|
||||||
|
import sys
|
||||||
|
import traceback
|
||||||
import typing
|
import typing
|
||||||
from urllib.parse import unquote
|
from urllib.parse import unquote
|
||||||
|
|
||||||
import flask
|
import flask
|
||||||
import requests
|
import requests
|
||||||
|
import werkzeug
|
||||||
|
from werkzeug.debug.tbtools import DebugTraceback
|
||||||
|
|
||||||
app = flask.Flask(__name__)
|
app = flask.Flask(__name__)
|
||||||
app.debug = True
|
app.debug = False
|
||||||
|
|
||||||
enwiki = "en.wikipedia.org/wiki/"
|
enwiki = "en.wikipedia.org/wiki/"
|
||||||
|
|
||||||
|
|
||||||
|
@app.errorhandler(werkzeug.exceptions.InternalServerError)
|
||||||
|
def exception_handler(e: werkzeug.exceptions.InternalServerError) -> tuple[str, int]:
|
||||||
|
"""Handle exception."""
|
||||||
|
exec_type, exc_value, current_traceback = sys.exc_info()
|
||||||
|
assert exc_value
|
||||||
|
tb = DebugTraceback(exc_value)
|
||||||
|
|
||||||
|
summary = tb.render_traceback_html(include_title=False)
|
||||||
|
exc_lines = "".join(tb._te.format_exception_only())
|
||||||
|
|
||||||
|
last_frame = list(traceback.walk_tb(current_traceback))[-1][0]
|
||||||
|
last_frame_args = inspect.getargs(last_frame.f_code)
|
||||||
|
|
||||||
|
return (
|
||||||
|
flask.render_template(
|
||||||
|
"show_error.html",
|
||||||
|
plaintext=tb.render_traceback_text(),
|
||||||
|
exception=exc_lines,
|
||||||
|
exception_type=tb._te.exc_type.__name__,
|
||||||
|
summary=summary,
|
||||||
|
last_frame=last_frame,
|
||||||
|
last_frame_args=last_frame_args,
|
||||||
|
),
|
||||||
|
500,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
@app.route("/")
|
@app.route("/")
|
||||||
def start() -> str:
|
def start() -> str:
|
||||||
"""Start form."""
|
"""Start form."""
|
||||||
|
|
78
static/css/exception.css
Normal file
78
static/css/exception.css
Normal file
|
@ -0,0 +1,78 @@
|
||||||
|
div.debugger { text-align: left; padding: 12px; margin: auto;
|
||||||
|
background-color: white; }
|
||||||
|
div.detail { cursor: pointer; }
|
||||||
|
div.detail p { margin: 0 0 8px 13px; font-size: 14px; white-space: pre-wrap;
|
||||||
|
font-family: monospace; }
|
||||||
|
div.explanation { margin: 20px 13px; font-size: 15px; color: #555; }
|
||||||
|
div.footer { font-size: 13px; text-align: right; margin: 30px 0;
|
||||||
|
color: #86989B; }
|
||||||
|
|
||||||
|
h2 { font-size: 16px; margin: 1.3em 0 0.0 0; padding: 9px;
|
||||||
|
background-color: #11557C; color: white; }
|
||||||
|
h2 em, h3 em { font-style: normal; color: #A5D6D9; font-weight: normal; }
|
||||||
|
|
||||||
|
div.traceback, div.plain { border: 1px solid #ddd; margin: 0 0 1em 0; padding: 10px; }
|
||||||
|
div.plain p { margin: 0; }
|
||||||
|
div.plain textarea,
|
||||||
|
div.plain pre { margin: 10px 0 0 0; padding: 4px;
|
||||||
|
background-color: #E8EFF0; border: 1px solid #D3E7E9; }
|
||||||
|
div.plain textarea { width: 99%; height: 300px; }
|
||||||
|
div.traceback h3 { font-size: 1em; margin: 0 0 0.8em 0; }
|
||||||
|
div.traceback ul { list-style: none; margin: 0; padding: 0 0 0 1em; }
|
||||||
|
div.traceback h4 { font-size: 13px; font-weight: normal; margin: 0.7em 0 0.1em 0; }
|
||||||
|
div.traceback pre { margin: 0; padding: 5px 0 3px 15px;
|
||||||
|
background-color: #E8EFF0; border: 1px solid #D3E7E9; }
|
||||||
|
div.traceback .library .current { background: white; color: #555; }
|
||||||
|
div.traceback .expanded .current { background: #E8EFF0; color: black; }
|
||||||
|
div.traceback pre:hover { background-color: #DDECEE; color: black; cursor: pointer; }
|
||||||
|
div.traceback div.source.expanded pre + pre { border-top: none; }
|
||||||
|
|
||||||
|
div.traceback span.ws { display: none; }
|
||||||
|
div.traceback pre.before, div.traceback pre.after { display: none; background: white; }
|
||||||
|
div.traceback div.source.expanded pre.before,
|
||||||
|
div.traceback div.source.expanded pre.after {
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
|
||||||
|
div.traceback div.source.expanded span.ws {
|
||||||
|
display: inline;
|
||||||
|
}
|
||||||
|
|
||||||
|
div.traceback blockquote { margin: 1em 0 0 0; padding: 0; white-space: pre-line; }
|
||||||
|
div.traceback img { float: right; padding: 2px; margin: -3px 2px 0 0; display: none; }
|
||||||
|
div.traceback img:hover { background-color: #ddd; cursor: pointer;
|
||||||
|
border-color: #BFDDE0; }
|
||||||
|
div.traceback pre:hover img { display: block; }
|
||||||
|
div.traceback cite.filename { font-style: normal; color: #3B666B; }
|
||||||
|
|
||||||
|
pre.console { border: 1px solid #ccc; background: white!important;
|
||||||
|
color: black; padding: 5px!important;
|
||||||
|
margin: 3px 0 0 0!important; cursor: default!important;
|
||||||
|
max-height: 400px; overflow: auto; }
|
||||||
|
pre.console form { color: #555; }
|
||||||
|
pre.console input { background-color: transparent; color: #555;
|
||||||
|
width: 90%; font-family: 'Consolas', 'Deja Vu Sans Mono',
|
||||||
|
'Bitstream Vera Sans Mono', monospace; font-size: 14px;
|
||||||
|
border: none!important; }
|
||||||
|
|
||||||
|
span.string { color: #30799B; }
|
||||||
|
span.number { color: #9C1A1C; }
|
||||||
|
span.help { color: #3A7734; }
|
||||||
|
span.object { color: #485F6E; }
|
||||||
|
span.extended { opacity: 0.5; }
|
||||||
|
span.extended:hover { opacity: 1; }
|
||||||
|
a.toggle { text-decoration: none; background-repeat: no-repeat;
|
||||||
|
background-position: center center;
|
||||||
|
background-image: url(?__debugger__=yes&cmd=resource&f=more.png); }
|
||||||
|
a.toggle:hover { background-color: #444; }
|
||||||
|
a.open { background-image: url(?__debugger__=yes&cmd=resource&f=less.png); }
|
||||||
|
|
||||||
|
div.traceback pre, div.console pre {
|
||||||
|
white-space: pre-wrap; /* css-3 should we be so lucky... */
|
||||||
|
white-space: -moz-pre-wrap; /* Mozilla, since 1999 */
|
||||||
|
white-space: -pre-wrap; /* Opera 4-6 ?? */
|
||||||
|
white-space: -o-pre-wrap; /* Opera 7 ?? */
|
||||||
|
word-wrap: break-word; /* Internet Explorer 5.5+ */
|
||||||
|
_white-space: pre; /* IE only hack to re-specify in
|
||||||
|
addition to word-wrap */
|
||||||
|
}
|
23
templates/show_error.html
Normal file
23
templates/show_error.html
Normal file
|
@ -0,0 +1,23 @@
|
||||||
|
{% extends "base.html" %}
|
||||||
|
|
||||||
|
{% block style %}
|
||||||
|
<link rel="stylesheet" href="{{url_for('static', filename='css/exception.css')}}" />
|
||||||
|
{% endblock %}
|
||||||
|
|
||||||
|
{% block content %}
|
||||||
|
<div class="p-2">
|
||||||
|
|
||||||
|
<h1>Software error: {{ exception_type }}</h1>
|
||||||
|
<div>
|
||||||
|
<pre>{{ exception }}</pre>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<h2 class="traceback">Traceback <em>(most recent call last)</em></h2>
|
||||||
|
{{ summary | safe }}
|
||||||
|
|
||||||
|
<p>Error in function "{{ last_frame.f_code.co_name }}": {{ last_frame_args | pprint }}</p>
|
||||||
|
<pre>{{ last_frame.f_locals | pprint }}</pre>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{% endblock %}
|
Loading…
Reference in a new issue