Skip to main content

Using Rules

Liquibase Linter has several dozen rules that you can turn on and configure to suit your project.

Turning on a rule

No rules are turned on by default, but most can be turned on simply by adding a key with a true value to the rules object in the config file:

{
"rules": {
"isolate-ddl-changes": true
}
}

The value can also be an options object:

{
"rules": {
"isolate-ddl-changes": {
"enabled": true
}
}
}

Options

All rules also support these standard options (other than enabled):

  • errorMessage - (string) override the default error message for this rule, which is output when the rule fails on a change. This can be useful if you are using a rule in a very targeted way and want to make it clear to the developer why it has failed. Most rules make the invalid value they found available to be interpolated with %s.
  • condition - (string) - Spring EL expression that should resolve to a boolean, which if provided will decide whether the rule should be applied or not. The expression scope is as follows -
    • DatabaseChangeLog object available as changeLog
    • ChangeSet object available as changeSet
    • Change object available as change
    • matchesContext helper function which can be used like matchesContext('foo', 'bar'). This function just delegates to the liquibase context matching method so the same logic applies.
  • enableAfter - (string) allows you to specify a change log file name after which this rule should be enabled. See Retrofitting for more detail.

Individual rules also support their own options; you can find these documented with those rules.

Multiple Configs

Though you might not need it often, you can specify multiple configs - with different options - for the same rule. You can do this by providing an array of rule config objects rather than just one, as in this example:

{
"rules": {
"object-name": [
{
"pattern": "^(?!_)[A-Z_0-9]+(?<!_)$",
"errorMessage": "Object name '%s' name must be uppercase and use '_' separation"
},
{
"pattern": "^POWER.*$",
"errorMessage": "Object name '%s' name must begin with 'POWER'"
}
]
}
}

If you provide multiple configs, each applicable change/changeset/changelog will be checked with all of the configs in turn. A failure on any of the configs will be treated as a failure - in other words, your scripts have to pass against all the configs, so the logic is "AND" rather than "OR".

Failure

Once a rule is switched on, it will be run against each of your scripts right after Liquibase parses them from their source format (e.g. XML). If a rule fails (that is, a script broke the rule) then Liquibase will exit with a ChangeLogParseException containing details of which change failed and why, and nothing will be run into the target database.