Localize.Exception behaviour (Localize v0.32.0)

Copy Markdown View Source

Conventions and a small behaviour shared by structured Localize exceptions.

The :reason field

Exceptions that distinguish between multiple failure categories carry a :reason field whose value is a documented atom from a closed set. Callers can pattern-match on :reason to branch on category without parsing the rendered message; message/1 is the single place a user-facing sentence is assembled.

Modules that adopt this convention declare @behaviour Localize.Exception and implement reason_atoms/0, returning the exhaustive list of valid :reason values for that struct. This lets tooling — including the structural-exception test suite — iterate the documented reasons and verify that message/1 has a clause for each.

The :cause field

When a higher layer must report a lower layer's failure, the outer exception carries the inner one in a :cause field of type Exception.t() | nil. The convention is:

  • :cause is set when, and only when, the outer exception is a wrapper. A direct error sets :cause to nil.

  • The outer message/1 may delegate to the inner via Exception.message(cause), possibly with a leading context phrase.

  • Programmatic callers can pattern-match on the outer struct for the operation context, and call Exception.message/1 on :cause for the original detail.

This convention is used by Localize.FormatError, Localize.LocaleDownloadError, and Localize.ParseError.

Why no prose in structural fields

Fields like :reason, :expected, :path, and :detail must hold structured values — atoms, paths, short labels, struct references — not free-form sentences with interpolated values. Putting a sentence in a structural field defeats pattern matching, duplicates content into message/1, and prevents translation.

Summary

Callbacks

Returns the closed set of atoms permitted in this exception's :reason field.

Callbacks

reason_atoms()

@callback reason_atoms() :: [atom()]

Returns the closed set of atoms permitted in this exception's :reason field.

Used by tests to verify that message/1 has a rendering clause for each documented reason atom. The list MUST be exhaustive — any atom assigned to :reason at runtime must appear here.