Skip to content

Credo check to catch and avoid unwanted exception silent swallows in Elixir code

License

Notifications You must be signed in to change notification settings

dgilperez/credo_exception_swallow

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

2 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

CredoExceptionSwallow

Hex.pm Hex.pm Hex.pm

A Credo check to detect silent exception swallowing in Elixir rescue blocks.

Silent exception handling is a dangerous anti-pattern that:

  • Hides bugs in production
  • Makes debugging extremely difficult
  • Creates unpredictable system behavior

This check enforces proper error handling by requiring that every rescue block either logs, reports to error monitoring, or re-raises the exception.

Installation

Add to your mix.exs:

def deps do
  [
    {:credo_exception_swallow, "~> 0.1.0", only: [:dev, :test], runtime: false}
  ]
end

Configuration

Add to your .credo.exs in the checks: %{enabled: [...]} section:

{CredoExceptionSwallow.Checks.Warning.SilentRescue, []}

Options

{CredoExceptionSwallow.Checks.Warning.SilentRescue, [
  # Exclude specific files (e.g., health checks)
  files: %{excluded: ["lib/my_app_web/controllers/health_controller.ex"]},
  # Set priority (:high, :normal, :low)
  priority: :high,
  # Skip test files (default: true)
  skip_test_files: true,
  # Additional acceptable function calls beyond defaults
  acceptable_calls: [
    "MyApp.ErrorHandler.report"
  ]
]}

What It Detects

Bad Examples (will trigger warning)

# Silent swallow - VERY BAD
try do
  risky_operation()
rescue
  _ -> nil
end

# Silent swallow with specific exception - STILL BAD
try do
  parse_data(input)
rescue
  ArgumentError -> {:error, :invalid}
end

Good Examples (acceptable patterns)

# Log the error
try do
  risky_operation()
rescue
  e ->
    Logger.error("Operation failed: #{inspect(e)}")
    {:error, :failed}
end

# Report to error monitoring
try do
  risky_operation()
rescue
  e ->
    Sentry.capture_exception(e, stacktrace: __STACKTRACE__)
    {:error, :failed}
end

# Re-raise (let it crash philosophy)
try do
  risky_operation()
rescue
  e -> reraise e, __STACKTRACE__
end

Default Acceptable Calls

The following function calls are considered proper error handling:

  • Logger.error/1,2
  • Logger.warning/1,2
  • Logger.warn/1,2
  • Logger.info/1,2
  • Logger.debug/1,2
  • ErrorReporter.report_exception/1,2
  • ErrorReporter.report_message/1,2
  • Sentry.capture_exception/1,2
  • Sentry.capture_message/1,2
  • reraise/2
  • raise/1,2

License

MIT License - see LICENSE file.

About

Credo check to catch and avoid unwanted exception silent swallows in Elixir code

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

 
 
 

Contributors

Languages