Skip to content
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

Support native JSON module #135

Open
wants to merge 3 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
31 changes: 30 additions & 1 deletion assets/js/app.js
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,36 @@ let socketPath = document.querySelector("meta[name='socket-path']").getAttribute
let socketTransport = document.querySelector("meta[name='socket-transport']").getAttribute("content");
let normalizedTransport = (socketTransport == "longpoll") ? LongPoll : WebSocket;

let liveSocket = new LiveSocket(socketPath, Socket, { transport: normalizedTransport, params: { _csrf_token: csrfToken }});

const Hooks = {
JsonPrettyPrint: {
mounted() {
this.formatJson();
},
updated() {
this.formatJson();
},
formatJson() {
try {
// Get the raw JSON content
const rawJson = this.el.textContent.trim();
// Parse and stringify with indentation
const formattedJson = JSON.stringify(JSON.parse(rawJson), null, 2);
// Update the element content
this.el.textContent = formattedJson;
} catch (error) {
console.error("Error formatting JSON:", error);
// Keep the original content if there's an error
}
}
}
};

let liveSocket = new LiveSocket(socketPath, Socket, {
transport: normalizedTransport,
params: { _csrf_token: csrfToken },
hooks: Hooks
});

// Show progress bar on live navigation and form submits
topbar.config({ barColors: { 0: "#29d" }, shadowColor: "rgba(0, 0, 0, .3)" });
Expand Down
20 changes: 14 additions & 6 deletions lib/error_tracker.ex
Original file line number Diff line number Diff line change
Expand Up @@ -223,14 +223,13 @@ defmodule ErrorTracker do

## Content serialization

The content stored on the context should be serializable using the JSON library
used by the application (usually `Jason`), so it is rather recommended to use
primitive types (strings, numbers, booleans...).
The content stored on the context should be serializable using the JSON library used by the
application (usually `JSON` for Elixir 1.18+ and `Jason` for older versions), so it is
recommended to use primitive types (strings, numbers, booleans...).

If you still need to pass more complex data types to your context, please test
that they can be encoded to JSON or storing the errors will fail. In the case
of `Jason` that may require defining an Encoder for that data type if not
included by default.
that they can be encoded to JSON or storing the errors will fail. You may need to define a
custom encoder for that data type if not included by default.
"""
@spec set_context(context()) :: context()
def set_context(params) when is_map(params) do
Expand Down Expand Up @@ -383,4 +382,13 @@ defmodule ErrorTracker do
Telemetry.new_occurrence(occurrence, muted)
occurrence
end

@doc false
def __default_json_encoder__ do
# Elixir 1.18+ includes the JSON module. On older versions we should fall back to Jason (which
# is listed as an optional dependency).
if Version.match?(System.version(), ">= 1.18.0"),
do: JSON,
else: Jason
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

👀

I think this may generate compile warnings. I tried compiling in v1.180 and simulating Jason is not available (changing it to something different, like Jason2) and it generates a compile warning even not being used in runtime.

And that may happen also in the other direction (trying to find JSON in Elixir < v1.17, but haven't tested that).

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is interesting. I will create two plain Phoenix projects (one with Jason and other without) and see how it behaves.

If this throws warnings we may need to ask in the forum how to handle it properly.

end
end
40 changes: 23 additions & 17 deletions lib/error_tracker/schemas/occurrence.ex
Original file line number Diff line number Diff line change
Expand Up @@ -46,26 +46,32 @@ defmodule ErrorTracker.Occurrence do
if changeset.valid? do
context = get_field(changeset, :context, %{})

json_encoder =
db_json_encoder =
ErrorTracker.Repo.with_adapter(fn
:postgres -> Application.get_env(:postgrex, :json_library, Jason)
:mysql -> Application.get_env(:myxql, :json_library, Jason)
:sqlite -> Application.get_env(:ecto_sqlite3, :json_library, Jason)
:postgres -> Application.get_env(:postgrex, :json_library)
:mysql -> Application.get_env(:myxql, :json_library)
:sqlite -> Application.get_env(:ecto_sqlite3, :json_library)
end)

case json_encoder.encode_to_iodata(context) do
{:ok, _} ->
put_change(changeset, :context, context)

{:error, _} ->
Logger.warning(
"[ErrorTracker] Context has been ignored: it is not serializable to JSON."
)

put_change(changeset, :context, %{
error: "Context not stored because it contains information not serializable to JSON."
})
end
validated_context =
try do
json_encoder = db_json_encoder || ErrorTracker.__default_json_encoder__()
_iodata = json_encoder.encode_to_iodata!(context)

context
rescue
_e in Protocol.UndefinedError ->
Logger.warning(
"[ErrorTracker] Context has been ignored: it is not serializable to JSON."
)

%{
error:
"Context not stored because it contains information not serializable to JSON."
}
end

put_change(changeset, :context, validated_context)
else
changeset
end
Expand Down
8 changes: 7 additions & 1 deletion lib/error_tracker/web/live/show.html.heex
Original file line number Diff line number Diff line change
Expand Up @@ -77,7 +77,13 @@
</.section>

<.section title="Context">
<pre class="overflow-auto text-sm p-4 rounded-lg bg-gray-300/10 border border-gray-900"><%= Jason.encode!(@occurrence.context, pretty: true) %></pre>
<pre
id="context"
class="overflow-auto text-sm p-4 rounded-lg bg-gray-300/10 border border-gray-900"
phx-hook="JsonPrettyPrint"
>
<%= ErrorTracker.__default_json_encoder__().encode_to_iodata!(@occurrence.context) %>
</pre>
</.section>
</div>

Expand Down
2 changes: 1 addition & 1 deletion mix.exs
Original file line number Diff line number Diff line change
Expand Up @@ -86,10 +86,10 @@ defmodule ErrorTracker.MixProject do
[
{:ecto_sql, "~> 3.0"},
{:ecto, "~> 3.11"},
{:jason, "~> 1.1"},
{:phoenix_live_view, "~> 0.19 or ~> 1.0"},
{:phoenix_ecto, "~> 4.6"},
{:plug, "~> 1.10"},
{:jason, "~> 1.1", optional: true},
{:postgrex, ">= 0.0.0", optional: true},
{:myxql, ">= 0.0.0", optional: true},
{:ecto_sqlite3, ">= 0.0.0", optional: true},
Expand Down
6,001 changes: 5,994 additions & 7 deletions priv/static/app.js

Large diffs are not rendered by default.

Loading