Skip to main content

prefer-set-or-object-rule

Summary: Prefer set or object rule over comprehension

Category: Idiomatic

Avoid

package policy

# top level set comprehension
developers := {developer |
some user in input.users
"developer" in user.roles
developer := user.name
}

# top level object comprehension
user_roles_mapping := {user: roles |
some user in input.users
roles := user.roles
}

Prefer

package policy

# set generating rule
developers contains developer if {
some user in input.users
"developer" in user.roles
developer := user.name
}

# object generating rule
user_roles_mapping[user] := roles if {
some user in input.users
roles := user.roles
}

Rationale

Comprehensions are awesome, and should be part of any policy author's toolbox. Using comprehensions inside of rule bodies allow for a wide variety of elegant solutions to otherwise hard problems. However, when used as the value directly (and unconditionally) assigned to a rule, it is almost always better to use a rule that generates a set or object in the rule body rather than having a comprehension do so in the rule head. Why is that?

Readability

Rules that generate objects, and sets even more so, read more natural than comprehensions, and are generally more descriptive. While both constructs are easy to spot for a seasoned Rego author, anything that helps improve readability is a win.

Extensibility

While readability is important, the real benefit of using a rule to generate a set or object is that it allows the rule to be extended. This is particularly true for set generating rules, and it's not by accident they often are referred to as multi-value rules. A rule assigned the value of a set comprehension can't have its value changed later, or more items added to the set. A set generating rule however, can easily be extended to contain more items, conditionally or unconditionally.

package policy

# Getting developers from input
developers contains developer if {
some user in input.users
"developer" in user.roles
developer := user.name
}

# *Also* getting developers from data
developers contains developer if {
some user in data.users
"developer" in user.roles
developer := user.name
}

# Unconditionally adding a developer to the set
developers contains "Hackerman"

In the example above, all three rules contribute to the developers set. If we wanted to, we could even create another policy file using the same package, and have more rules added there that would contribute to the set. This creates some great opportunities for extensibility, and collaboration across developers and teams working on policy together.

Objects differ somewhat from sets in that while several rules can be used to generate an object, there cannot be more than one rule contributing to a single key-value pair.

package policy

novels[title] := content if {
some document in input.documents
document.type == "novel"
title := document.title
content := document.content
}

# This works as long as "The Hobbit" is not already in the novels object
novels["The Hobbit"] := "In a hole in the ground there lived a hobbit."

# Map and set generating objects can also be combined, in which case the
# value is extensible even for the same key! In the example above, more
# rules could help contribute titles to an author, perhaps using different
# data sources.
titles_by_author[document.author] contains document.title if {
some document in input.documents
}

Exceptions

Note that this rule does not apply to array comprehensions, as there is no equivalent tp use a rule to generate an array.

This rule will also ignore simple comprehensions used solely for the purpose of converting an array to a set, i.e:

package policy

# Convert set to array. This is fine.
my_set := {item | some item in arr}

Configuration Options

This linter rule provides the following configuration options:

rules:
idiomatic:
prefer-set-or-object-rule:
# one of "error", "warning", "ignore"
level: error

Community

If you think you've found a problem with this rule or its documentation, would like to suggest improvements, new rules, or just talk about Regal in general, please join us in the #regal channel in the Styra Community Slack!