Curated library of 38 atomic skills, 7 personas, and 1 orchestrator for Elixir and Phoenix development. Organized by category: fundamentals, phoenix, database, testing, auth, infrastructure, quality, security, integrations, tooling, frameworks, personas, and orchestration. Covers core Elixir patterns, Phoenix LiveView, Ecto, OTP, Oban, testing, security, deployment, real-time, and modern tooling (Req, Swoosh, Cachex, Broadway, Ash).
73
91%
Does it follow best practices?
Impact
—
No eval scenarios have been run
Advisory
Suggest reviewing before use
Use this skill before writing ANY telemetry, logging, or metrics code.
Logger.info("action", key: value)) — never string interpolation in log messagesApplication.start/2 — not in modules that may restart, and never in GenServer initEcto.Repo telemetry events for query monitoring — Ecto already emits events; don't manually instrument queriesPhoenix.LiveDashboard in dev/staging — free observability with zero code:debug level in production — it includes query parameters and PII; use :info level instead❌ Bad — unsearchable:
Logger.info("User #{user.id} created order #{order.id} for $#{order.total}")✅ Good — searchable, parseable:
Logger.info("Order created", user_id: user.id, order_id: order.id, total: order.total)# In a Plug
defmodule MyAppWeb.Plugs.RequestMetadata do
import Plug.Conn
def call(conn, _opts) do
Logger.metadata(
request_id: conn.assigns[:request_id] || Ecto.UUID.generate(),
remote_ip: to_string(:inet.ntoa(conn.remote_ip))
)
conn
end
end# Emit a custom event
:telemetry.execute(
[:my_app, :orders, :created], # event name
%{count: 1, total_cents: 4999}, # measurements
%{user_id: user.id, source: :web} # metadata
)✅ Good — in application.ex:
defmodule MyApp.Application do
use Application
@impl true
def start(_type, _args) do
MyApp.Telemetry.attach_handlers()
children = [MyApp.Repo, MyAppWeb.Endpoint]
Supervisor.start_link(children, strategy: :one_for_one)
end
end
defmodule MyApp.Telemetry do
require Logger
def attach_handlers do
:telemetry.attach_many("my-app-handlers", [
[:my_app, :orders, :created],
[:my_app, :payments, :processed]
], &handle_event/4, nil)
end
def handle_event([:my_app, :orders, :created], measurements, metadata, _config) do
Logger.info("Order created",
total_cents: measurements.count,
user_id: metadata.user_id
)
end
endAfter attaching handlers, confirm everything is wired up in iex:
# In iex -S mix
:telemetry.execute([:my_app, :orders, :created], %{count: 1, total_cents: 4999}, %{user_id: 42, source: :web})
# Expected: Logger output like — [info] Order created total_cents=1 user_id=42If no log line appears, verify attach_handlers/0 was called before the event and that the event name matches exactly.
def process_order(order) do
:telemetry.span([:my_app, :orders, :process], %{order_id: order.id}, fn ->
result = do_process(order)
{result, %{order_id: order.id, status: :completed}}
end)
enddef handle_slow_query(_event, measurements, metadata, %{threshold_ms: threshold}) do
duration_ms = System.convert_time_unit(measurements.total_time, :native, :millisecond)
if duration_ms > threshold do
Logger.warning("Slow query",
duration_ms: duration_ms,
source: metadata.source,
query: metadata.query
)
end
end# router.ex
import Phoenix.LiveDashboard.Router
scope "/" do
pipe_through :browser
live_dashboard "/dashboard",
metrics: MyAppWeb.Telemetry,
ecto_repos: [MyApp.Repo]
endDeeper topics: For performance profiling and benchmarking, see the
benchee-profilingskill. For exporting metrics to external tools (Prometheus, Datadog) and production deployment considerations, see thedeployment-gotchasskill. For low-level metric aggregation and reporter configuration, consult the Telemetry.Metrics and TelemetryMetricsPrometheus library docs.
| Predecessor | This Skill | Successor |
|---|---|---|
| otp-essentials | telemetry-essentials | deployment-gotchas |
| otp-essentials | telemetry-essentials | benchee-profiling |
| security-essentials | telemetry-essentials | deployment-gotchas |