-
-
Notifications
You must be signed in to change notification settings - Fork 12
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
feat(dev_server): display compiler errors #120
Conversation
This is amazing! I'll take a closer look in the next couple days. This might be a useful thing to extract to https://github.com/elixir-tools/web_dev_utils but that can come later. |
Hi! i was looking into your PR the other week, and had some thoughts. first is, this doesn't actually show compiler errors for you project, only runtime errors once tableau attempts to build your site secondly, with regard to the runtime error, it presents a confusing error, as its wrapped in tableau related code. I dont think this is necessarily at the fault of the code you've written, its more on tableau doesn't currently show more understandable errors when it fails, they're usually wrapped in tasks and are more hidden in the stack trace. while I was investigating, I remembered that Plug already ships with Plug.Debugger to present views like this. I think all that is missing is to add that plug, and patch |
What I was testing were the following patches in tableau and web_dev_utils respectively diff --git a/lib/tableau_dev_server/router.ex b/lib/tableau_dev_server/router.ex
index 699d042..9bccdac 100644
--- a/lib/tableau_dev_server/router.ex
+++ b/lib/tableau_dev_server/router.ex
@@ -1,6 +1,7 @@
defmodule TableauDevServer.Router do
@moduledoc false
use Plug.Router, init_mode: :runtime
+ use Plug.Debugger
require Logger
@@ -32,11 +33,19 @@ defmodule TableauDevServer.Router do
end
defp recompile(conn, _) do
- if conn.request_path != "/ws" do
- WebDevUtils.CodeReloader.reload()
+ if conn.request_path == "/ws" do
+ conn
+ else
+ case WebDevUtils.CodeReloader.reload() do
+ {:error, [error | _rest]} ->
+ with %{stacktrace: stacktrace, details: {:error, %exception{} = error}} <- error do
+ reraise exception, error |> Map.from_struct() |> Enum.to_list(), stacktrace
+ end
+
+ _ ->
+ conn
+ end
end
-
- conn
end
defp rerender(conn, _) do diff --git a/lib/web_dev_utils/code_reloader.ex b/lib/web_dev_utils/code_reloader.ex
index 7a023ed..912f613 100644
--- a/lib/web_dev_utils/code_reloader.ex
+++ b/lib/web_dev_utils/code_reloader.ex
@@ -25,12 +25,12 @@ defmodule WebDevUtils.CodeReloader do
{:ok, :nostate}
end
- def handle_call(:reload, from, state) do
- froms = all_waiting([from])
- mix_compile(Code.ensure_loaded(Mix.Task))
- Enum.each(froms, &GenServer.reply(&1, :ok))
+ def handle_call(:reload, _from, state) do
+ froms = all_waiting([])
+ result = mix_compile(Code.ensure_loaded(Mix.Task))
+ Enum.each(froms, &GenServer.reply(&1, result))
- {:noreply, state}
+ {:reply, result, state}
end
defp all_waiting(acc) do
@@ -43,6 +43,7 @@ defmodule WebDevUtils.CodeReloader do
defp mix_compile({:error, _reason}) do
Logger.error("Could not find Mix")
+ :error
end
defp mix_compile({:module, Mix.Task}) do |
Hi, thanks for getting back to this. I only added I tried to improve it a bit and added a change that wraps the exception and adds information about the page it occurred on. Maybe it's a bit better than showing the error itself, what do you think? diff --git a/lib/mix/tasks/tableau.build.ex b/lib/mix/tasks/tableau.build.ex
index 4d03860..3c19cae 100644
--- a/lib/mix/tasks/tableau.build.ex
+++ b/lib/mix/tasks/tableau.build.ex
@@ -44,12 +44,17 @@ defmodule Mix.Tasks.Tableau.Build do
pages =
pages
|> Task.async_stream(fn {mod, page} ->
- content = Tableau.Document.render(graph, mod, token, page)
- permalink = Nodable.permalink(mod)
-
- Map.merge(page, %{body: content, permalink: permalink})
+ try do
+ content = Tableau.Document.render(graph, mod, token, page)
+ permalink = Nodable.permalink(mod)
+
+ Map.merge(page, %{body: content, permalink: permalink})
+ rescue
+ exception ->
+ reraise Tableau.BuildException, [page: page, exception: exception], __STACKTRACE__
+ end
end)
- |> Stream.map(fn {:ok, result} -> result end)
+ |> Stream.map(&map_pages/1)
|> Enum.to_list()
token = put_in(token.site[:pages], pages)
@@ -74,6 +79,12 @@ defmodule Mix.Tasks.Tableau.Build do
token
end
+ defp map_pages({:ok, result}), do: result
+
+ defp map_pages({:exit, {exception, stacktrace}}) do
+ reraise exception, stacktrace
+ end
+
@file_extensions [".html"]
defp build_file_path(out, permalink) do
if Path.extname(permalink) in @file_extensions do
diff --git a/lib/tableau.ex b/lib/tableau.ex
index 6cc2449..1f0b77f 100644
--- a/lib/tableau.ex
+++ b/lib/tableau.ex
@@ -51,3 +51,29 @@ defmodule Tableau do
MDEx.to_html!(content, Keyword.merge(config.markdown[:mdex], overrides))
end
end
+
+defmodule Tableau.BuildException do
+ @moduledoc false
+ @type t :: %__MODULE__{
+ page: map(),
+ exception: any()
+ }
+
+ defexception [:page, :exception]
+
+ @impl true
+ def exception(page: page, exception: exception) do
+ %__MODULE__{page: page, exception: exception}
+ end
+
+ @impl true
+ def message(%__MODULE__{page: page, exception: exception}) do
+ """
+ An exception was raised:
+ #{Exception.format_banner(:error, exception)}
+
+ occurred during the build process on the page:
+ #{inspect(page, pretty: true)}
+ """
+ end
+end
diff --git a/lib/tableau_dev_server/router.ex b/lib/tableau_dev_server/router.ex
index 699d042..d791994 100644
--- a/lib/tableau_dev_server/router.ex
+++ b/lib/tableau_dev_server/router.ex
@@ -1,6 +1,7 @@
defmodule TableauDevServer.Router do
@moduledoc false
use Plug.Router, init_mode: :runtime
+ use Plug.Debugger
require Logger
|
Maybe not, but you've confirmed it works with a compiler error? I think the diff you've commented looks good, feel free to push it up and i'll test it out myself, I want to see what it looks like in the terminal as well |
ae73368
to
547e1c7
Compare
|
This is what the patch I commented with did. I'll get the web dev utils patch released and we can update tableau and move get this merged |
Now I'm thinking... tableau needs a logo to customize the error page with! Haha 😆 |
I released v0.3.0 of web dev utils, should unblock you |
lib/mix/tasks/tableau.build.ex
Outdated
defp map_pages({:ok, result}), do: result | ||
|
||
defp map_pages({:exit, {exception, stacktrace}}) do | ||
reraise exception, stacktrace | ||
end | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
let's inline this function
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ok, I updated the code
{:error, errors} -> | ||
errors = Enum.filter(errors, &(&1.severity == :error)) | ||
message = Enum.reduce(errors, "", fn error, acc -> "#{acc}\n#{error.message}" end) | ||
stacktrace = List.first(errors).stacktrace |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
this can fail if the list is is only non error severity, which would make List.first return nil
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It should not fail as we are expecting error, first value in tuple is :error
.
For the :ok
it might return Mix.Task.Compiler.Diagnostic
with warnings.
lib/tableau_dev_server/router.ex
Outdated
case WebDevUtils.CodeReloader.reload() do | ||
{:error, errors} -> | ||
errors = Enum.filter(errors, &(&1.severity == :error)) | ||
message = Enum.reduce(errors, "", fn error, acc -> "#{acc}\n#{error.message}" end) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think you just want Enum.map_join here
Enum.map_join(errors, "\n", & &1.message)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ok, I updated the code
Tested locally, pretty sweet. I added some code to strip the ANSI escape sequences from the compiler errors. Thanks a lot! |
Inspired by Phoenix’s code reloading, this PR enhances Tableau’s error handling by displaying compiler errors directly in the browser.

Let me know if you'd like any modifications!