123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600 |
- import codecs
- import inspect
- import os
- import re
- import sys
- import sysconfig
- import traceback
- import typing as t
- from html import escape
- from tokenize import TokenError
- from types import CodeType
- from types import TracebackType
- from .._internal import _to_str
- from ..filesystem import get_filesystem_encoding
- from ..utils import cached_property
- from .console import Console
- _coding_re = re.compile(br"coding[:=]\s*([-\w.]+)")
- _line_re = re.compile(br"^(.*?)$", re.MULTILINE)
- _funcdef_re = re.compile(r"^(\s*def\s)|(.*(?<!\w)lambda(:|\s))|^(\s*@)")
- HEADER = """\
- <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"
- "http://www.w3.org/TR/html4/loose.dtd">
- <html>
- <head>
- <title>%(title)s // Werkzeug Debugger</title>
- <link rel="stylesheet" href="?__debugger__=yes&cmd=resource&f=style.css"
- type="text/css">
- <!-- We need to make sure this has a favicon so that the debugger does
- not accidentally trigger a request to /favicon.ico which might
- change the application's state. -->
- <link rel="shortcut icon"
- href="?__debugger__=yes&cmd=resource&f=console.png">
- <script src="?__debugger__=yes&cmd=resource&f=debugger.js"></script>
- <script type="text/javascript">
- var TRACEBACK = %(traceback_id)d,
- CONSOLE_MODE = %(console)s,
- EVALEX = %(evalex)s,
- EVALEX_TRUSTED = %(evalex_trusted)s,
- SECRET = "%(secret)s";
- </script>
- </head>
- <body style="background-color: #fff">
- <div class="debugger">
- """
- FOOTER = """\
- <div class="footer">
- Brought to you by <strong class="arthur">DON'T PANIC</strong>, your
- friendly Werkzeug powered traceback interpreter.
- </div>
- </div>
- <div class="pin-prompt">
- <div class="inner">
- <h3>Console Locked</h3>
- <p>
- The console is locked and needs to be unlocked by entering the PIN.
- You can find the PIN printed out on the standard output of your
- shell that runs the server.
- <form>
- <p>PIN:
- <input type=text name=pin size=14>
- <input type=submit name=btn value="Confirm Pin">
- </form>
- </div>
- </div>
- </body>
- </html>
- """
- PAGE_HTML = (
- HEADER
- + """\
- <h1>%(exception_type)s</h1>
- <div class="detail">
- <p class="errormsg">%(exception)s</p>
- </div>
- <h2 class="traceback">Traceback <em>(most recent call last)</em></h2>
- %(summary)s
- <div class="plain">
- <p>
- This is the Copy/Paste friendly version of the traceback.
- </p>
- <textarea cols="50" rows="10" name="code" readonly>%(plaintext)s</textarea>
- </div>
- <div class="explanation">
- The debugger caught an exception in your WSGI application. You can now
- look at the traceback which led to the error. <span class="nojavascript">
- If you enable JavaScript you can also use additional features such as code
- execution (if the evalex feature is enabled), automatic pasting of the
- exceptions and much more.</span>
- </div>
- """
- + FOOTER
- + """
- <!--
- %(plaintext_cs)s
- -->
- """
- )
- CONSOLE_HTML = (
- HEADER
- + """\
- <h1>Interactive Console</h1>
- <div class="explanation">
- In this console you can execute Python expressions in the context of the
- application. The initial namespace was created by the debugger automatically.
- </div>
- <div class="console"><div class="inner">The Console requires JavaScript.</div></div>
- """
- + FOOTER
- )
- SUMMARY_HTML = """\
- <div class="%(classes)s">
- %(title)s
- <ul>%(frames)s</ul>
- %(description)s
- </div>
- """
- FRAME_HTML = """\
- <div class="frame" id="frame-%(id)d">
- <h4>File <cite class="filename">"%(filename)s"</cite>,
- line <em class="line">%(lineno)s</em>,
- in <code class="function">%(function_name)s</code></h4>
- <div class="source %(library)s">%(lines)s</div>
- </div>
- """
- SOURCE_LINE_HTML = """\
- <tr class="%(classes)s">
- <td class=lineno>%(lineno)s</td>
- <td>%(code)s</td>
- </tr>
- """
- def render_console_html(secret: str, evalex_trusted: bool = True) -> str:
- return CONSOLE_HTML % {
- "evalex": "true",
- "evalex_trusted": "true" if evalex_trusted else "false",
- "console": "true",
- "title": "Console",
- "secret": secret,
- "traceback_id": -1,
- }
- def get_current_traceback(
- ignore_system_exceptions: bool = False,
- show_hidden_frames: bool = False,
- skip: int = 0,
- ) -> "Traceback":
- """Get the current exception info as `Traceback` object. Per default
- calling this method will reraise system exceptions such as generator exit,
- system exit or others. This behavior can be disabled by passing `False`
- to the function as first parameter.
- """
- info = t.cast(
- t.Tuple[t.Type[BaseException], BaseException, TracebackType], sys.exc_info()
- )
- exc_type, exc_value, tb = info
- if ignore_system_exceptions and exc_type in {
- SystemExit,
- KeyboardInterrupt,
- GeneratorExit,
- }:
- raise
- for _ in range(skip):
- if tb.tb_next is None:
- break
- tb = tb.tb_next
- tb = Traceback(exc_type, exc_value, tb)
- if not show_hidden_frames:
- tb.filter_hidden_frames()
- return tb
- class Line:
- """Helper for the source renderer."""
- __slots__ = ("lineno", "code", "in_frame", "current")
- def __init__(self, lineno: int, code: str) -> None:
- self.lineno = lineno
- self.code = code
- self.in_frame = False
- self.current = False
- @property
- def classes(self) -> t.List[str]:
- rv = ["line"]
- if self.in_frame:
- rv.append("in-frame")
- if self.current:
- rv.append("current")
- return rv
- def render(self) -> str:
- return SOURCE_LINE_HTML % {
- "classes": " ".join(self.classes),
- "lineno": self.lineno,
- "code": escape(self.code),
- }
- class Traceback:
- """Wraps a traceback."""
- def __init__(
- self,
- exc_type: t.Type[BaseException],
- exc_value: BaseException,
- tb: TracebackType,
- ) -> None:
- self.exc_type = exc_type
- self.exc_value = exc_value
- self.tb = tb
- exception_type = exc_type.__name__
- if exc_type.__module__ not in {"builtins", "__builtin__", "exceptions"}:
- exception_type = f"{exc_type.__module__}.{exception_type}"
- self.exception_type = exception_type
- self.groups = []
- memo = set()
- while True:
- self.groups.append(Group(exc_type, exc_value, tb))
- memo.add(id(exc_value))
- exc_value = exc_value.__cause__ or exc_value.__context__ # type: ignore
- if exc_value is None or id(exc_value) in memo:
- break
- exc_type = type(exc_value)
- tb = exc_value.__traceback__ # type: ignore
- self.groups.reverse()
- self.frames = [frame for group in self.groups for frame in group.frames]
- def filter_hidden_frames(self) -> None:
- """Remove the frames according to the paste spec."""
- for group in self.groups:
- group.filter_hidden_frames()
- self.frames[:] = [frame for group in self.groups for frame in group.frames]
- @property
- def is_syntax_error(self) -> bool:
- """Is it a syntax error?"""
- return isinstance(self.exc_value, SyntaxError)
- @property
- def exception(self) -> str:
- """String representation of the final exception."""
- return self.groups[-1].exception
- def log(self, logfile: t.Optional[t.IO[str]] = None) -> None:
- """Log the ASCII traceback into a file object."""
- if logfile is None:
- logfile = sys.stderr
- tb = f"{self.plaintext.rstrip()}\n"
- logfile.write(tb)
- def render_summary(self, include_title: bool = True) -> str:
- """Render the traceback for the interactive console."""
- title = ""
- classes = ["traceback"]
- if not self.frames:
- classes.append("noframe-traceback")
- frames = []
- else:
- library_frames = sum(frame.is_library for frame in self.frames)
- mark_lib = 0 < library_frames < len(self.frames)
- frames = [group.render(mark_lib=mark_lib) for group in self.groups]
- if include_title:
- if self.is_syntax_error:
- title = "Syntax Error"
- else:
- title = "Traceback <em>(most recent call last)</em>:"
- if self.is_syntax_error:
- description = f"<pre class=syntaxerror>{escape(self.exception)}</pre>"
- else:
- description = f"<blockquote>{escape(self.exception)}</blockquote>"
- return SUMMARY_HTML % {
- "classes": " ".join(classes),
- "title": f"<h3>{title if title else ''}</h3>",
- "frames": "\n".join(frames),
- "description": description,
- }
- def render_full(
- self,
- evalex: bool = False,
- secret: t.Optional[str] = None,
- evalex_trusted: bool = True,
- ) -> str:
- """Render the Full HTML page with the traceback info."""
- exc = escape(self.exception)
- return PAGE_HTML % {
- "evalex": "true" if evalex else "false",
- "evalex_trusted": "true" if evalex_trusted else "false",
- "console": "false",
- "title": exc,
- "exception": exc,
- "exception_type": escape(self.exception_type),
- "summary": self.render_summary(include_title=False),
- "plaintext": escape(self.plaintext),
- "plaintext_cs": re.sub("-{2,}", "-", self.plaintext),
- "traceback_id": self.id,
- "secret": secret,
- }
- @cached_property
- def plaintext(self) -> str:
- return "\n".join([group.render_text() for group in self.groups])
- @property
- def id(self) -> int:
- return id(self)
- class Group:
- """A group of frames for an exception in a traceback. If the
- exception has a ``__cause__`` or ``__context__``, there are multiple
- exception groups.
- """
- def __init__(
- self,
- exc_type: t.Type[BaseException],
- exc_value: BaseException,
- tb: TracebackType,
- ) -> None:
- self.exc_type = exc_type
- self.exc_value = exc_value
- self.info = None
- if exc_value.__cause__ is not None:
- self.info = (
- "The above exception was the direct cause of the following exception"
- )
- elif exc_value.__context__ is not None:
- self.info = (
- "During handling of the above exception, another exception occurred"
- )
- self.frames = []
- while tb is not None:
- self.frames.append(Frame(exc_type, exc_value, tb))
- tb = tb.tb_next # type: ignore
- def filter_hidden_frames(self) -> None:
- # An exception may not have a traceback to filter frames, such
- # as one re-raised from ProcessPoolExecutor.
- if not self.frames:
- return
- new_frames: t.List[Frame] = []
- hidden = False
- for frame in self.frames:
- hide = frame.hide
- if hide in ("before", "before_and_this"):
- new_frames = []
- hidden = False
- if hide == "before_and_this":
- continue
- elif hide in ("reset", "reset_and_this"):
- hidden = False
- if hide == "reset_and_this":
- continue
- elif hide in ("after", "after_and_this"):
- hidden = True
- if hide == "after_and_this":
- continue
- elif hide or hidden:
- continue
- new_frames.append(frame)
- # if we only have one frame and that frame is from the codeop
- # module, remove it.
- if len(new_frames) == 1 and self.frames[0].module == "codeop":
- del self.frames[:]
- # if the last frame is missing something went terrible wrong :(
- elif self.frames[-1] in new_frames:
- self.frames[:] = new_frames
- @property
- def exception(self) -> str:
- """String representation of the exception."""
- buf = traceback.format_exception_only(self.exc_type, self.exc_value)
- rv = "".join(buf).strip()
- return _to_str(rv, "utf-8", "replace")
- def render(self, mark_lib: bool = True) -> str:
- out = []
- if self.info is not None:
- out.append(f'<li><div class="exc-divider">{self.info}:</div>')
- for frame in self.frames:
- title = f' title="{escape(frame.info)}"' if frame.info else ""
- out.append(f"<li{title}>{frame.render(mark_lib=mark_lib)}")
- return "\n".join(out)
- def render_text(self) -> str:
- out = []
- if self.info is not None:
- out.append(f"\n{self.info}:\n")
- out.append("Traceback (most recent call last):")
- for frame in self.frames:
- out.append(frame.render_text())
- out.append(self.exception)
- return "\n".join(out)
- class Frame:
- """A single frame in a traceback."""
- def __init__(
- self,
- exc_type: t.Type[BaseException],
- exc_value: BaseException,
- tb: TracebackType,
- ) -> None:
- self.lineno = tb.tb_lineno
- self.function_name = tb.tb_frame.f_code.co_name
- self.locals = tb.tb_frame.f_locals
- self.globals = tb.tb_frame.f_globals
- fn = inspect.getsourcefile(tb) or inspect.getfile(tb)
- if fn[-4:] in (".pyo", ".pyc"):
- fn = fn[:-1]
- # if it's a file on the file system resolve the real filename.
- if os.path.isfile(fn):
- fn = os.path.realpath(fn)
- self.filename = _to_str(fn, get_filesystem_encoding())
- self.module = self.globals.get("__name__", self.locals.get("__name__"))
- self.loader = self.globals.get("__loader__", self.locals.get("__loader__"))
- self.code = tb.tb_frame.f_code
- # support for paste's traceback extensions
- self.hide = self.locals.get("__traceback_hide__", False)
- info = self.locals.get("__traceback_info__")
- if info is not None:
- info = _to_str(info, "utf-8", "replace")
- self.info = info
- def render(self, mark_lib: bool = True) -> str:
- """Render a single frame in a traceback."""
- return FRAME_HTML % {
- "id": self.id,
- "filename": escape(self.filename),
- "lineno": self.lineno,
- "function_name": escape(self.function_name),
- "lines": self.render_line_context(),
- "library": "library" if mark_lib and self.is_library else "",
- }
- @cached_property
- def is_library(self) -> bool:
- return any(
- self.filename.startswith(os.path.realpath(path))
- for path in sysconfig.get_paths().values()
- )
- def render_text(self) -> str:
- return (
- f' File "{self.filename}", line {self.lineno}, in {self.function_name}\n'
- f" {self.current_line.strip()}"
- )
- def render_line_context(self) -> str:
- before, current, after = self.get_context_lines()
- rv = []
- def render_line(line: str, cls: str) -> None:
- line = line.expandtabs().rstrip()
- stripped_line = line.strip()
- prefix = len(line) - len(stripped_line)
- rv.append(
- f'<pre class="line {cls}"><span class="ws">{" " * prefix}</span>'
- f"{escape(stripped_line) if stripped_line else ' '}</pre>"
- )
- for line in before:
- render_line(line, "before")
- render_line(current, "current")
- for line in after:
- render_line(line, "after")
- return "\n".join(rv)
- def get_annotated_lines(self) -> t.List[Line]:
- """Helper function that returns lines with extra information."""
- lines = [Line(idx + 1, x) for idx, x in enumerate(self.sourcelines)]
- # find function definition and mark lines
- if hasattr(self.code, "co_firstlineno"):
- lineno = self.code.co_firstlineno - 1
- while lineno > 0:
- if _funcdef_re.match(lines[lineno].code):
- break
- lineno -= 1
- try:
- offset = len(inspect.getblock([f"{x.code}\n" for x in lines[lineno:]]))
- except TokenError:
- offset = 0
- for line in lines[lineno : lineno + offset]:
- line.in_frame = True
- # mark current line
- try:
- lines[self.lineno - 1].current = True
- except IndexError:
- pass
- return lines
- def eval(self, code: t.Union[str, CodeType], mode: str = "single") -> t.Any:
- """Evaluate code in the context of the frame."""
- if isinstance(code, str):
- code = compile(code, "<interactive>", mode)
- return eval(code, self.globals, self.locals)
- @cached_property
- def sourcelines(self) -> t.List[str]:
- """The sourcecode of the file as list of strings."""
- # get sourcecode from loader or file
- source = None
- if self.loader is not None:
- try:
- if hasattr(self.loader, "get_source"):
- source = self.loader.get_source(self.module)
- elif hasattr(self.loader, "get_source_by_code"):
- source = self.loader.get_source_by_code(self.code)
- except Exception:
- # we munch the exception so that we don't cause troubles
- # if the loader is broken.
- pass
- if source is None:
- try:
- with open(self.filename, mode="rb") as f:
- source = f.read()
- except OSError:
- return []
- # already str? return right away
- if isinstance(source, str):
- return source.splitlines()
- charset = "utf-8"
- if source.startswith(codecs.BOM_UTF8):
- source = source[3:]
- else:
- for idx, match in enumerate(_line_re.finditer(source)):
- coding_match = _coding_re.search(match.group())
- if coding_match is not None:
- charset = coding_match.group(1).decode("utf-8")
- break
- if idx > 1:
- break
- # on broken cookies we fall back to utf-8 too
- charset = _to_str(charset)
- try:
- codecs.lookup(charset)
- except LookupError:
- charset = "utf-8"
- return source.decode(charset, "replace").splitlines()
- def get_context_lines(
- self, context: int = 5
- ) -> t.Tuple[t.List[str], str, t.List[str]]:
- before = self.sourcelines[self.lineno - context - 1 : self.lineno - 1]
- past = self.sourcelines[self.lineno : self.lineno + context]
- return (before, self.current_line, past)
- @property
- def current_line(self) -> str:
- try:
- return self.sourcelines[self.lineno - 1]
- except IndexError:
- return ""
- @cached_property
- def console(self) -> Console:
- return Console(self.globals, self.locals)
- @property
- def id(self) -> int:
- return id(self)
|