zf

zenflows testing
git clone https://s.sonu.ch/~srfsh/zf.git
Log | Files | Refs | Submodules | README | LICENSE

debugger.ex (28319B)


      1 defmodule Plug.Debugger do
      2   @moduledoc """
      3   A module (**not a plug**) for debugging in development.
      4 
      5   This module is commonly used within a `Plug.Builder` or a `Plug.Router`
      6   and it wraps the `call/2` function.
      7 
      8   Notice `Plug.Debugger` *does not* catch errors, as errors should still
      9   propagate so that the Elixir process finishes with the proper reason.
     10   This module does not perform any logging either, as all logging is done
     11   by the web server handler.
     12 
     13   **Note:** If this module is used with `Plug.ErrorHandler`, only one of
     14   them will effectively handle errors. For this reason, it is recommended
     15   that `Plug.Debugger` is used before `Plug.ErrorHandler` and only in
     16   particular environments, like `:dev`.
     17 
     18   In case of an error, the rendered page drops the `content-security-policy`
     19   header before rendering the error to ensure that the error is displayed
     20   correctly.
     21 
     22   ## Examples
     23 
     24       defmodule MyApp do
     25         use Plug.Builder
     26 
     27         if Mix.env == :dev do
     28           use Plug.Debugger, otp_app: :my_app
     29         end
     30 
     31         plug :boom
     32 
     33         def boom(conn, _) do
     34           # Error raised here will be caught and displayed in a debug page
     35           # complete with a stacktrace and other helpful info.
     36           raise "oops"
     37         end
     38       end
     39 
     40   ## Options
     41 
     42     * `:otp_app` - the OTP application that is using Plug. This option is used
     43       to filter stacktraces that belong only to the given application.
     44     * `:style` - custom styles (see below)
     45     * `:banner` - the optional MFA (`{module, function, args}`) which receives
     46       exception details and returns banner contents to appear at the top of
     47       the page. May be any string, including markup.
     48 
     49   ## Custom styles
     50 
     51   You may pass a `:style` option to customize the look of the HTML page.
     52 
     53       use Plug.Debugger, style:
     54         [primary: "#c0392b", logo: "data:image/png;base64,..."]
     55 
     56   The following keys are available:
     57 
     58     * `:primary` - primary color
     59     * `:accent` - accent color
     60     * `:logo` - logo URI, or `nil` to disable
     61 
     62   The `:logo` is preferred to be a base64-encoded data URI so not to make any
     63   external requests, though external URLs (eg, `https://...`) are supported.
     64 
     65   ## Custom Banners
     66 
     67   You may pass an MFA (`{module, function, args}`) to be invoked when an
     68   error is rendered which provides a custom banner at the top of the
     69   debugger page. The function receives the following arguments, with the
     70   passed `args` concatenated at the end:
     71 
     72       [conn, status, kind, reason, stacktrace]
     73 
     74   For example, the following `:banner` option:
     75 
     76       use Plug.Debugger, banner: {MyModule, :debug_banner, []}
     77 
     78   would invoke the function:
     79 
     80       MyModule.debug_banner(conn, status, kind, reason, stacktrace)
     81 
     82   ## Links to the text editor
     83 
     84   If a `PLUG_EDITOR` environment variable is set, `Plug.Debugger` will
     85   use it to generate links to your text editor. The variable should be
     86   set with `__FILE__` and `__LINE__` placeholders which will be correctly
     87   replaced. For example (with the [TextMate](http://macromates.com) editor):
     88 
     89       txmt://open/?url=file://__FILE__&line=__LINE__
     90 
     91   Or, using Visual Studio Code:
     92 
     93       vscode://file/__FILE__:__LINE__
     94   """
     95 
     96   @already_sent {:plug_conn, :sent}
     97 
     98   @logo ""
     99 
    100   @default_style %{
    101     primary: "#4e2a8e",
    102     accent: "#607080",
    103     highlight: "#f0f4fa",
    104     red_highlight: "#ffe5e5",
    105     line_color: "#eee",
    106     text_color: "#203040",
    107     logo: @logo,
    108     monospace_font: "menlo, consolas, monospace"
    109   }
    110 
    111   @salt "plug-debugger-actions"
    112 
    113   import Plug.Conn
    114   require Logger
    115 
    116   @doc false
    117   defmacro __using__(opts) do
    118     quote do
    119       @plug_debugger unquote(opts)
    120       @before_compile Plug.Debugger
    121     end
    122   end
    123 
    124   @doc false
    125   defmacro __before_compile__(_) do
    126     quote location: :keep do
    127       defoverridable call: 2
    128 
    129       def call(conn, opts) do
    130         try do
    131           case conn do
    132             %Plug.Conn{path_info: ["__plug__", "debugger", "action"], method: "POST"} ->
    133               Plug.Debugger.run_action(conn)
    134 
    135             %Plug.Conn{} ->
    136               super(conn, opts)
    137           end
    138         rescue
    139           e in Plug.Conn.WrapperError ->
    140             %{conn: conn, kind: kind, reason: reason, stack: stack} = e
    141             Plug.Debugger.__catch__(conn, kind, reason, stack, @plug_debugger)
    142         catch
    143           kind, reason ->
    144             Plug.Debugger.__catch__(conn, kind, reason, __STACKTRACE__, @plug_debugger)
    145         end
    146       end
    147     end
    148   end
    149 
    150   @doc false
    151   def __catch__(conn, kind, reason, stack, opts) do
    152     reason = Exception.normalize(kind, reason, stack)
    153     status = status(kind, reason)
    154 
    155     receive do
    156       @already_sent ->
    157         send(self(), @already_sent)
    158         log(status, kind, reason, stack)
    159         :erlang.raise(kind, reason, stack)
    160     after
    161       0 ->
    162         render(conn, status, kind, reason, stack, opts)
    163         log(status, kind, reason, stack)
    164         :erlang.raise(kind, reason, stack)
    165     end
    166   end
    167 
    168   # We don't log status >= 500 because those are treated as errors and logged later.
    169   defp log(status, kind, reason, stack) when status < 500,
    170     do: Logger.debug(Exception.format(kind, reason, stack))
    171 
    172   defp log(_status, _kind, _reason, _stack), do: :ok
    173 
    174   ## Rendering
    175 
    176   require EEx
    177 
    178   html_template_path = "lib/plug/templates/debugger.html.eex"
    179   EEx.function_from_file(:defp, :template_html, html_template_path, [:assigns])
    180 
    181   markdown_template_path = "lib/plug/templates/debugger.md.eex"
    182   EEx.function_from_file(:defp, :template_markdown, markdown_template_path, [:assigns])
    183 
    184   # Made public with @doc false for testing.
    185   @doc false
    186   def render(conn, status, kind, reason, stack, opts) do
    187     session = maybe_fetch_session(conn)
    188     params = maybe_fetch_query_params(conn)
    189     {title, message} = info(kind, reason)
    190 
    191     assigns = [
    192       conn: conn,
    193       title: title,
    194       formatted: Exception.format(kind, reason, stack),
    195       session: session,
    196       params: params,
    197       frames: frames(:md, stack, opts)
    198     ]
    199 
    200     markdown = template_markdown(assigns)
    201 
    202     if accepts_html?(get_req_header(conn, "accept")) do
    203       conn =
    204         conn
    205         |> put_resp_content_type("text/html")
    206         |> delete_resp_header("content-security-policy")
    207 
    208       actions = encoded_actions_for_exception(reason, conn)
    209       last_path = actions_redirect_path(conn)
    210       style = Enum.into(opts[:style] || [], @default_style)
    211       banner = banner(conn, status, kind, reason, stack, opts)
    212 
    213       assigns =
    214         Keyword.merge(assigns,
    215           conn: conn,
    216           message: message,
    217           markdown: markdown,
    218           style: style,
    219           banner: banner,
    220           actions: actions,
    221           frames: frames(:html, stack, opts),
    222           last_path: last_path
    223         )
    224 
    225       send_resp(conn, status, template_html(assigns))
    226     else
    227       conn = put_resp_content_type(conn, "text/markdown")
    228       send_resp(conn, status, markdown)
    229     end
    230   end
    231 
    232   @doc false
    233   def run_action(%Plug.Conn{} = conn) do
    234     with %Plug.Conn{body_params: params} <- fetch_body_params(conn),
    235          {:ok, {module, function, args}} <-
    236            Plug.Crypto.verify(conn.secret_key_base, @salt, params["encoded_handler"]) do
    237       apply(module, function, args)
    238 
    239       conn
    240       |> Plug.Conn.put_resp_header("location", params["last_path"] || "/")
    241       |> send_resp(302, "")
    242       |> halt()
    243     else
    244       _ -> raise "could not run Plug.Debugger action"
    245     end
    246   end
    247 
    248   @doc false
    249   def encoded_actions_for_exception(exception, conn) do
    250     exception_implementation = Plug.Exception.impl_for(exception)
    251 
    252     implements_actions? =
    253       Code.ensure_loaded?(exception_implementation) &&
    254         function_exported?(exception_implementation, :actions, 1)
    255 
    256     # TODO: Remove implements_actions? in future Plug versions
    257     if implements_actions? && conn.secret_key_base do
    258       actions = Plug.Exception.actions(exception)
    259 
    260       Enum.map(actions, fn %{label: label, handler: handler} ->
    261         encoded_handler = Plug.Crypto.sign(conn.secret_key_base, @salt, handler)
    262         %{label: label, encoded_handler: encoded_handler}
    263       end)
    264     else
    265       []
    266     end
    267   end
    268 
    269   defp actions_redirect_path(%Plug.Conn{
    270          method: "GET",
    271          request_path: request_path,
    272          query_string: query_string
    273        }) do
    274     case query_string do
    275       "" -> request_path
    276       query_string -> "#{request_path}?#{query_string}"
    277     end
    278   end
    279 
    280   defp actions_redirect_path(conn) do
    281     case get_req_header(conn, "referer") do
    282       [referer] -> referer
    283       [] -> "/"
    284     end
    285   end
    286 
    287   defp accepts_html?(_accept_header = []), do: false
    288 
    289   defp accepts_html?(_accept_header = [header | _]),
    290     do: String.contains?(header, ["*/*", "text/*", "text/html"])
    291 
    292   defp maybe_fetch_session(conn) do
    293     if conn.private[:plug_session_fetch] do
    294       conn |> fetch_session(conn) |> get_session()
    295     end
    296   end
    297 
    298   defp maybe_fetch_query_params(conn) do
    299     fetch_query_params(conn).params
    300   rescue
    301     Plug.Conn.InvalidQueryError ->
    302       case conn.params do
    303         %Plug.Conn.Unfetched{} -> %{}
    304         params -> params
    305       end
    306   end
    307 
    308   @parsers_opts Plug.Parsers.init(parsers: [:urlencoded])
    309   defp fetch_body_params(conn), do: Plug.Parsers.call(conn, @parsers_opts)
    310 
    311   defp status(:error, error), do: Plug.Exception.status(error)
    312   defp status(_, _), do: 500
    313 
    314   defp info(:error, error), do: {inspect(error.__struct__), Exception.message(error)}
    315   defp info(:throw, thrown), do: {"unhandled throw", inspect(thrown)}
    316   defp info(:exit, reason), do: {"unhandled exit", Exception.format_exit(reason)}
    317 
    318   defp frames(renderer, stacktrace, opts) do
    319     app = opts[:otp_app]
    320     editor = System.get_env("PLUG_EDITOR")
    321 
    322     stacktrace
    323     |> Enum.map_reduce(0, &each_frame(&1, &2, renderer, app, editor))
    324     |> elem(0)
    325   end
    326 
    327   defp each_frame(entry, index, renderer, root, editor) do
    328     {module, info, location, app, fun, arity, args} = get_entry(entry)
    329     {file, line} = {to_string(location[:file] || "nofile"), location[:line]}
    330 
    331     doc = module && get_doc(module, fun, arity, app)
    332     clauses = module && get_clauses(renderer, module, fun, args)
    333     source = get_source(app, module, file)
    334     context = get_context(root, app)
    335     snippet = get_snippet(source, line)
    336 
    337     {%{
    338        app: app,
    339        info: info,
    340        file: file,
    341        line: line,
    342        context: context,
    343        snippet: snippet,
    344        index: index,
    345        doc: doc,
    346        clauses: clauses,
    347        args: args,
    348        link: editor && get_editor(source, line, editor)
    349      }, index + 1}
    350   end
    351 
    352   # From :elixir_compiler_*
    353   defp get_entry({module, :__MODULE__, 0, location}) do
    354     {module, inspect(module) <> " (module)", location, get_app(module), nil, nil, nil}
    355   end
    356 
    357   # From :elixir_compiler_*
    358   defp get_entry({_module, :__MODULE__, 1, location}) do
    359     {nil, "(module)", location, nil, nil, nil, nil}
    360   end
    361 
    362   # From :elixir_compiler_*
    363   defp get_entry({_module, :__FILE__, 1, location}) do
    364     {nil, "(file)", location, nil, nil, nil, nil}
    365   end
    366 
    367   defp get_entry({module, fun, args, location}) when is_list(args) do
    368     arity = length(args)
    369     formatted_mfa = Exception.format_mfa(module, fun, arity)
    370     {module, formatted_mfa, location, get_app(module), fun, arity, args}
    371   end
    372 
    373   defp get_entry({module, fun, arity, location}) do
    374     {module, Exception.format_mfa(module, fun, arity), location, get_app(module), fun, arity, nil}
    375   end
    376 
    377   defp get_entry({fun, arity, location}) do
    378     {nil, Exception.format_fa(fun, arity), location, nil, fun, arity, nil}
    379   end
    380 
    381   defp get_app(module) do
    382     case :application.get_application(module) do
    383       {:ok, app} -> app
    384       :undefined -> nil
    385     end
    386   end
    387 
    388   defp get_doc(module, fun, arity, app) do
    389     with true <- has_docs?(module, fun, arity),
    390          {:ok, vsn} <- :application.get_key(app, :vsn) do
    391       vsn = vsn |> List.to_string() |> String.split("-") |> hd()
    392       fun = fun |> Atom.to_string() |> URI.encode()
    393       "https://hexdocs.pm/#{app}/#{vsn}/#{inspect(module)}.html##{fun}/#{arity}"
    394     else
    395       _ -> nil
    396     end
    397   end
    398 
    399   defp has_docs?(module, name, arity) do
    400     case Code.fetch_docs(module) do
    401       {:docs_v1, _, _, _, module_doc, _, docs} when module_doc != :hidden ->
    402         Enum.any?(docs, has_doc_matcher?(name, arity))
    403 
    404       _ ->
    405         false
    406     end
    407   end
    408 
    409   defp has_doc_matcher?(name, arity) do
    410     &match?(
    411       {{kind, ^name, ^arity}, _, _, doc, _}
    412       when kind in [:function, :macro] and doc != :hidden and doc != :none,
    413       &1
    414     )
    415   end
    416 
    417   defp get_clauses(renderer, module, fun, args) do
    418     with true <- is_list(args),
    419          {:ok, kind, clauses} <- Exception.blame_mfa(module, fun, args) do
    420       top_10 =
    421         clauses
    422         |> Enum.take(10)
    423         |> Enum.map(fn {args, guards} ->
    424           args = Enum.map_join(args, ", ", &blame_match(renderer, &1))
    425           base = "#{kind} #{fun}(#{args})"
    426           Enum.reduce(guards, base, &"#{&2} when #{blame_clause(renderer, &1)}")
    427         end)
    428 
    429       {length(top_10), length(clauses), top_10}
    430     else
    431       _ -> nil
    432     end
    433   end
    434 
    435   defp blame_match(:html, %{match?: true, node: node}),
    436     do: ~s(<i class="green">) <> h(Macro.to_string(node)) <> "</i>"
    437 
    438   defp blame_match(:html, %{match?: false, node: node}),
    439     do: ~s(<i class="red">) <> h(Macro.to_string(node)) <> "</i>"
    440 
    441   defp blame_match(_md, %{node: node}),
    442     do: h(Macro.to_string(node))
    443 
    444   defp blame_clause(renderer, {op, _, [left, right]}),
    445     do: blame_clause(renderer, left) <> " #{op} " <> blame_clause(renderer, right)
    446 
    447   defp blame_clause(renderer, node), do: blame_match(renderer, node)
    448 
    449   defp get_context(app, app) when app != nil, do: :app
    450   defp get_context(_app1, _app2), do: :all
    451 
    452   defp get_source(app, module, file) do
    453     cond do
    454       File.regular?(file) ->
    455         file
    456 
    457       File.regular?("apps/#{app}/#{file}") ->
    458         "apps/#{app}/#{file}"
    459 
    460       source = module && Code.ensure_loaded?(module) && module.module_info(:compile)[:source] ->
    461         to_string(source)
    462 
    463       true ->
    464         file
    465     end
    466   end
    467 
    468   defp get_editor(file, line, editor) do
    469     editor
    470     |> :binary.replace("__FILE__", URI.encode(Path.expand(file)))
    471     |> :binary.replace("__LINE__", to_string(line))
    472     |> h
    473   end
    474 
    475   @radius 5
    476 
    477   defp get_snippet(file, line) do
    478     if File.regular?(file) and is_integer(line) do
    479       to_discard = max(line - @radius - 1, 0)
    480       lines = File.stream!(file) |> Stream.take(line + 5) |> Stream.drop(to_discard)
    481 
    482       {first_five, lines} = Enum.split(lines, line - to_discard - 1)
    483       first_five = with_line_number(first_five, to_discard + 1, false)
    484 
    485       {center, last_five} = Enum.split(lines, 1)
    486       center = with_line_number(center, line, true)
    487       last_five = with_line_number(last_five, line + 1, false)
    488 
    489       first_five ++ center ++ last_five
    490     end
    491   end
    492 
    493   defp with_line_number(lines, initial, highlight) do
    494     lines
    495     |> Enum.map_reduce(initial, fn line, acc -> {{acc, line, highlight}, acc + 1} end)
    496     |> elem(0)
    497   end
    498 
    499   defp banner(conn, status, kind, reason, stack, opts) do
    500     case Keyword.fetch(opts, :banner) do
    501       {:ok, {mod, func, args}} ->
    502         apply(mod, func, [conn, status, kind, reason, stack] ++ args)
    503 
    504       {:ok, other} ->
    505         raise ArgumentError,
    506               "expected :banner to be an MFA ({module, func, args}), got: #{inspect(other)}"
    507 
    508       :error ->
    509         nil
    510     end
    511   end
    512 
    513   ## Helpers
    514 
    515   defp method(%Plug.Conn{method: method}), do: method
    516 
    517   defp url(%Plug.Conn{scheme: scheme, host: host, port: port} = conn),
    518     do: "#{scheme}://#{host}:#{port}#{conn.request_path}"
    519 
    520   defp h(string) do
    521     string |> to_string() |> Plug.HTML.html_escape()
    522   end
    523 end