How audits work
An audit is a SELECT query that returns rows that violate a condition. Zero rows means the audit passes. Any rows returned means a failure. Forerror severity audits:
- Full table builds: SQLBuild materializes into a staging table, runs audits against it, and only promotes to the target if all audits pass. If any fail, the staging table is kept for inspection and the production table is untouched.
- Incremental models: Delta-phase audits validate each batch before DML is applied. If an audit fails, the batch is not applied.
warn severity audits, the build continues and the failure is reported in the output.
Built-in audits
SQLBuild includes four generic audits out of the box. You do not need to define these inaudits/generic/ - they are available automatically:
| Audit | Description | Parameters |
|---|---|---|
not_null | Fails if any row has a NULL value in the column | Column-level only |
unique | Fails if any non-NULL value appears more than once | Column-level only |
accepted_values | Fails if any non-NULL value is not in the allowed list | values - list of allowed values |
relationships | Fails if any non-NULL value does not exist in the referenced column | to - target relation, field - target column |
Using built-in audits
Attach them in theMODEL() header like any generic audit:
Overriding built-in audits
If you define a generic audit with the same name as a built-in (e.g.audits/generic/not_null.sql), your definition takes precedence. SQLBuild emits a warning so you’re aware of the override:
Custom generic audits
Beyond the built-ins, you can define reusable SQL templates underaudits/generic/. They use @parameter placeholders that are resolved by the audit engine at compile time.
Audit parameters
Generic audit SQL uses@name for parameter placeholders. These are resolved by the audit engine, not the general SQL interpolation system:
| Parameter | Description |
|---|---|
@column | The column name (auto-populated for column-level audits) |
@relation | The target relation (auto-populated from the attached model or source) |
@'name' | A quoted parameter passed from the audit declaration (e.g. @'values') |
@name | An unquoted parameter passed from the audit declaration (e.g. @expression) |
Attaching custom generic audits
Singular audits
Singular audits are standalone SQL files underaudits/ (outside the generic/ directory) that reference models directly. They’re useful for one-off checks that don’t fit a reusable template.
__ref() calls in the query. If the audit references a single model, it attaches to that model. If it references multiple models, SQLBuild attaches it to the latest (most downstream) model in the DAG. If attachment can’t be inferred, the audit runs at the end of the build.
Source audits
Sources support the same audit system as models. Audits attached to sources run before any dependent model is built:error severity fails, all downstream models that depend on that source are blocked. This lets you catch data quality issues at the source before any transformations run.
Severity
| Severity | Behavior |
|---|---|
error | Blocks the build. Staging table is not promoted, DML is not applied. |
warn | Reports a warning but allows the build to continue. |
sqlbuild_project.toml:
MODEL() header:
Run scope
Audits on incremental models can run at different lifecycle phases:| Scope | Behavior |
|---|---|
final | Run once against the staged table before promotion (default). |
delta_and_final | Run against each delta batch before DML, then again against the target after all batches complete. |
error severity block DML before the target is updated. This is visible in the build output as audit (d) for delta-phase and audit (f) for final-phase:
4/4 indicates the audit passed for all 4 microbatch batches.
If a model is not incremental, delta_and_final degrades to final automatically.

