Skip to content

Commit

Permalink
Rego v1 capabilities and keywords update (#7216)
Browse files Browse the repository at this point in the history
* Separating v0- and v1 keywords
* Adding `rego_v1` capability feature

Signed-off-by: Johan Fylling <[email protected]>
  • Loading branch information
johanfylling authored Dec 17, 2024
1 parent 82b2214 commit 563321d
Show file tree
Hide file tree
Showing 38 changed files with 7,252 additions and 317 deletions.
10 changes: 5 additions & 5 deletions .github/workflows/pull-request.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -352,23 +352,23 @@ jobs:
version: edge

- name: Test policies
run: opa test build/policy
run: opa test --v0-compatible build/policy

- name: Ensure proper formatting
run: opa fmt --list --fail build/policy
run: opa fmt --v0-compatible --list --fail build/policy

- name: Run file policy checks on changed files
run: |
curl --silent --fail --header 'Authorization: Bearer ${{ secrets.GITHUB_TOKEN }}' -o files.json \
https://api.github.com/repos/${{ github.repository }}/pulls/${{ github.event.pull_request.number }}/files
opa eval -d build/policy/files.rego -d build/policy/helpers.rego --format values --input files.json \
opa eval --v0-compatible -d build/policy/files.rego -d build/policy/helpers.rego --format values --input files.json \
--fail-defined 'data.files.deny[message]'
env:
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}

- name: Show input on failure
run: opa eval --input files.json --format pretty input
run: opa eval --v0-compatible --input files.json --format pretty input
if: ${{ failure() }}

- name: Setup Hugo
Expand All @@ -386,4 +386,4 @@ jobs:
cd docs
make dev-generate hugo-production-build
cd -
opa eval 'data.integrations.deny[message]' -i docs/website/public/index.json -d build/policy/integrations.rego --format=values --fail-defined
opa eval --v0-compatible 'data.integrations.deny[message]' -i docs/website/public/index.json -d build/policy/integrations.rego --format=values --fail-defined
3 changes: 2 additions & 1 deletion ast/capabilities.go
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@ type VersionIndex = v1.VersionIndex
// heads, they wouldn't be able to parse them.
const FeatureRefHeadStringPrefixes = v1.FeatureRefHeadStringPrefixes
const FeatureRefHeads = v1.FeatureRefHeads
const FeatureRegoV1 = v1.FeatureRegoV1
const FeatureRegoV1Import = v1.FeatureRegoV1Import

// Capabilities defines a structure containing data that describes the capabilities
Expand All @@ -33,7 +34,7 @@ type WasmABIVersion = v1.WasmABIVersion

// CapabilitiesForThisVersion returns the capabilities of this version of OPA.
func CapabilitiesForThisVersion() *Capabilities {
return v1.CapabilitiesForThisVersion()
return v1.CapabilitiesForThisVersion(v1.CapabilitiesRegoVersion(DefaultRegoVersion))
}

// LoadCapabilitiesJSON loads a JSON serialized capabilities structure from the reader r.
Expand Down
4 changes: 4 additions & 0 deletions ast/parser.go
Original file line number Diff line number Diff line change
Expand Up @@ -43,3 +43,7 @@ type ParserOptions = v1.ParserOptions
func NewParser() *Parser {
return v1.NewParser().WithRegoVersion(DefaultRegoVersion)
}

func IsFutureKeyword(s string) bool {
return v1.IsFutureKeywordForRegoVersion(s, RegoV0)
}
4 changes: 1 addition & 3 deletions build/policy/files.rego
Original file line number Diff line number Diff line change
Expand Up @@ -7,9 +7,7 @@

package files

import future.keywords.contains
import future.keywords.if
import future.keywords.in
import rego.v1

import data.helpers.basename
import data.helpers.directory
Expand Down
10 changes: 5 additions & 5 deletions build/policy/files_test.rego
Original file line number Diff line number Diff line change
@@ -1,27 +1,27 @@
package files_test

import future.keywords.in
import rego.v1

import data.files.deny

test_deny_invalid_yaml_file {
test_deny_invalid_yaml_file if {
expected := "invalid.yaml is an invalid YAML file: {null{}}"
expected in deny with data.files.yaml_file_contents as {"invalid.yaml": "{null{}}"}
with data.files.changes as {"invalid.yaml": {"status": "modified"}}
}

test_allow_valid_yaml_file {
test_allow_valid_yaml_file if {
count(deny) == 0 with data.files.yaml_file_contents as {"valid.yaml": "foo: bar"}
with data.files.changes as {"valid.yaml": {"status": "modified"}}
}

test_deny_invalid_json_file {
test_deny_invalid_json_file if {
expected := "invalid.json is an invalid JSON file: }}}"
expected in deny with data.files.json_file_contents as {"invalid.json": "}}}"}
with data.files.changes as {"invalid.json": {"status": "modified"}}
}

test_allow_valid_json_file {
test_allow_valid_json_file if {
count(deny) == 0 with data.files.json_file_contents as {"valid.json": "{\"foo\": \"bar\"}"}
with data.files.changes as {"valid.json": {"status": "modified"}}
}
3 changes: 1 addition & 2 deletions build/policy/helpers.rego
Original file line number Diff line number Diff line change
@@ -1,7 +1,6 @@
package helpers

import future.keywords.if
import future.keywords.in
import rego.v1

last_indexof(string, search) := i if {
all := [i | chars := split(string, ""); chars[i] == search]
Expand Down
4 changes: 1 addition & 3 deletions build/policy/integrations.rego
Original file line number Diff line number Diff line change
@@ -1,8 +1,6 @@
package integrations

import future.keywords.contains
import future.keywords.if
import future.keywords.in
import rego.v1

allowed_image_extensions := ["png", "svg"]

Expand Down
60 changes: 30 additions & 30 deletions build/policy/integrations_test.rego
Original file line number Diff line number Diff line change
@@ -1,8 +1,8 @@
package integrations_test

import future.keywords.in
import rego.v1

messages_for_key(key, output) = messages {
messages_for_key(key, output) = messages if {
messages := {m |
some e
output[e]
Expand All @@ -15,18 +15,18 @@ messages_for_key(key, output) = messages {

print_if(true, _, _, _) = true

print_if(false, key, false, output) := false {
print_if(false, key, false, output) := false if {
print("Exp:", {})
print("Got: ", messages_for_key(key, output))
}

print_if(false, key, expected, output) := false {
print_if(false, key, expected, output) := false if {
is_string(expected)
print("Exp:", expected)
print("Got:", messages_for_key(key, output))
}

test_integration_has_valid_key {
test_integration_has_valid_key if {
output := data.integrations.deny with input as {"integrations": {"/integrations/in.valid/": {"link": "https://example.com/", "title": "Example"}}}

key := "key"
Expand All @@ -40,7 +40,7 @@ test_integration_has_valid_key {
print_if(result, key, message, output)
}

test_integration_has_required_fields_missing {
test_integration_has_required_fields_missing if {
output := data.integrations.deny with input as {"integrations": {"/integrations/regal/": {}}}

key := "fields"
Expand All @@ -53,7 +53,7 @@ test_integration_has_required_fields_missing {
print_if(result, key, message, output)
}

test_integration_has_required_fields_present {
test_integration_has_required_fields_present if {
output := data.integrations.deny with input as {"integrations": {"/integrations/regal/": {"title": "Regal"}}}

key := "fields"
Expand All @@ -66,7 +66,7 @@ test_integration_has_required_fields_present {
print_if(result, key, false, output)
}

test_integration_has_content_missing {
test_integration_has_content_missing if {
output := data.integrations.deny with input as {"integrations": {"/integrations/regal/": {}}}

key := "content"
Expand All @@ -79,7 +79,7 @@ test_integration_has_content_missing {
print_if(result, key, message, output)
}

test_integration_has_content_blank {
test_integration_has_content_blank if {
output := data.integrations.deny with input as {"integrations": {"/integrations/regal/": {"content": "\t\t\n "}}}

key := "content"
Expand All @@ -92,7 +92,7 @@ test_integration_has_content_blank {
print_if(result, key, message, output)
}

test_integration_has_content_present {
test_integration_has_content_present if {
output := data.integrations.deny with input as {"integrations": {"/integrations/regal/": {"content": "foobar"}}}

key := "content"
Expand All @@ -103,7 +103,7 @@ test_integration_has_content_present {
print_if(result, key, false, output)
}

test_every_integration_has_image_missing {
test_every_integration_has_image_missing if {
output := data.integrations.deny with input as {
"images": ["reegal.png"],
"integrations": {"/integrations/regal/": {}},
Expand All @@ -119,7 +119,7 @@ test_every_integration_has_image_missing {
print_if(result, key, message, output)
}

test_every_integration_has_image_present {
test_every_integration_has_image_present if {
output := data.integrations.deny with input as {
"images": ["regal.png"],
"integrations": {"regal": {}},
Expand All @@ -133,7 +133,7 @@ test_every_integration_has_image_present {
print_if(result, key, false, output)
}

test_every_integration_has_image_missing_but_permitted {
test_every_integration_has_image_missing_but_permitted if {
output := data.integrations.deny with input as {
"images": ["reegal.png"],
"integrations": {"regal": {"allow_missing_image": true}},
Expand All @@ -148,7 +148,7 @@ test_every_integration_has_image_missing_but_permitted {
print_if(result, key, false, output)
}

test_every_image_has_integration_missing {
test_every_image_has_integration_missing if {
output := data.integrations.deny with input as {
"images": ["regal.png"],
"integrations": {"foobar": {}},
Expand All @@ -164,7 +164,7 @@ test_every_image_has_integration_missing {
print_if(result, key, message, output)
}

test_every_image_has_integration_present {
test_every_image_has_integration_present if {
output := data.integrations.deny with input as {
"images": ["regal.png"],
"integrations": {"/integrations/regal/": {}},
Expand All @@ -178,7 +178,7 @@ test_every_image_has_integration_present {
print_if(result, key, false, output)
}

test_integration_organizations_missing {
test_integration_organizations_missing if {
output := data.integrations.deny with input as {
"organizations": {"/organizations/stira/": {}},
"integrations": {"/integrations/regal/": {"inventors": ["styra"]}},
Expand All @@ -194,7 +194,7 @@ test_integration_organizations_missing {
print_if(result, key, message, output)
}

test_integration_organizations_present {
test_integration_organizations_present if {
output := data.integrations.deny with input as {
"organizations": {"/organizations/styra/": {}},
"integrations": {"/integrations/regal/": {"inventors": ["styra"]}},
Expand All @@ -208,7 +208,7 @@ test_integration_organizations_present {
print_if(result, key, false, output)
}

test_integration_softwares_missing {
test_integration_softwares_missing if {
output := data.integrations.deny with input as {
"softwares": {"/softwares/mars/": {}},
"integrations": {"/integrations/regal/": {"software": ["terraform"]}},
Expand All @@ -224,7 +224,7 @@ test_integration_softwares_missing {
print_if(result, key, message, output)
}

test_integration_softwares_present {
test_integration_softwares_present if {
output := data.integrations.deny with input as {
"softwares": {"/softwares/terraform/": {}},
"integrations": {"/integrations/regal/": {"software": ["terraform"]}},
Expand All @@ -238,7 +238,7 @@ test_integration_softwares_present {
print_if(result, key, false, output)
}

test_software_has_required_fields_missing {
test_software_has_required_fields_missing if {
output := data.integrations.deny with input as {"softwares": {"/softwares/terraform/": {}}}

key := "fields"
Expand All @@ -251,7 +251,7 @@ test_software_has_required_fields_missing {
print_if(result, key, message, output)
}

test_software_has_required_fields_present {
test_software_has_required_fields_present if {
output := data.integrations.deny with input as {"softwares": {"terraform": {"link": "https://www.terraform.io/", "title": "Terraform"}}}

key := "fields"
Expand All @@ -263,7 +263,7 @@ test_software_has_required_fields_present {
print_if(result, key, false, output)
}

test_organization_has_required_labels {
test_organization_has_required_labels if {
output := data.integrations.deny with input as {"organizations": {"/organizations/styra/": {}}}

key := "fields"
Expand All @@ -276,7 +276,7 @@ test_organization_has_required_labels {
print_if(result, key, message, output)
}

test_organization_has_required_fields_present {
test_organization_has_required_fields_present if {
output := data.integrations.deny with input as {"organizations": {"styra": {"link": "https://styra.com/", "title": "Styra"}}}

key := "fields"
Expand All @@ -288,7 +288,7 @@ test_organization_has_required_fields_present {
print_if(result, key, false, output)
}

test_organization_has_valid_key {
test_organization_has_valid_key if {
output := data.integrations.deny with input as {"organizations": {"/organizations/sty.ra/": {"link": "https://styra.com/", "title": "Styra"}}}

key := "key"
Expand All @@ -302,7 +302,7 @@ test_organization_has_valid_key {
print_if(result, key, message, output)
}

test_organization_has_one_or_more_integrations_none {
test_organization_has_one_or_more_integrations_none if {
output := data.integrations.deny with input as {"organizations": {"/organizations/foobar/": {}}, "integrations": {}}

key := "orphaned_org"
Expand All @@ -315,7 +315,7 @@ test_organization_has_one_or_more_integrations_none {
print_if(result, key, message, output)
}

test_organization_has_one_or_more_integrations_one {
test_organization_has_one_or_more_integrations_one if {
output := data.integrations.deny with input as {"organizations": {"/organizations/foobaz/": {}}, "integrations": {"/integrations/foobar/": {"inventors": ["foobaz"]}}}

key := "orphaned_org"
Expand All @@ -326,7 +326,7 @@ test_organization_has_one_or_more_integrations_one {
print_if(result, key, false, output)
}

test_organization_has_one_or_more_integrations_speaker {
test_organization_has_one_or_more_integrations_speaker if {
output := data.integrations.deny with input as {"organizations": {"foobaz": {}}, "integrations": {"foobar": {"videos": [{"speakers": [{"organization": "foobaz"}]}]}}}

key := "orphaned_org"
Expand All @@ -337,7 +337,7 @@ test_organization_has_one_or_more_integrations_speaker {
print_if(result, key, false, output)
}

test_software_has_one_or_more_integrations_none {
test_software_has_one_or_more_integrations_none if {
output := data.integrations.deny with input as {"softwares": {"/softwares/foobar/": {}}, "integrations": {}}

key := "orphaned_software"
Expand All @@ -350,7 +350,7 @@ test_software_has_one_or_more_integrations_none {
print_if(result, key, message, output)
}

test_software_has_one_or_more_integrations_one {
test_software_has_one_or_more_integrations_one if {
output := data.integrations.deny with input as {"softwares": {"foobaz": {}}, "integrations": {"foobar": {"software": ["foobaz"]}}}

key := "orphaned_software"
Expand All @@ -361,7 +361,7 @@ test_software_has_one_or_more_integrations_one {
print_if(result, key, false, output)
}

test_software_has_valid_key {
test_software_has_valid_key if {
output := data.integrations.deny with input as {"softwares": {"/softwares/in.valid/": {"link": "https://example.com/", "title": "Example"}}}

key := "key"
Expand Down
Loading

0 comments on commit 563321d

Please sign in to comment.