diff --git a/pkg/kyverno/test/test_command.go b/pkg/kyverno/test/test_command.go index 94979db0b1..0a39fc1cac 100644 --- a/pkg/kyverno/test/test_command.go +++ b/pkg/kyverno/test/test_command.go @@ -38,98 +38,95 @@ import ( ) var longHelp = ` -The test command provides a facility to test resources against policies by comparing expected results, declared ahead of time in a test.yaml file, to actual results reported by Kyverno. Users provide the path to the folder containing a test.yaml file where the location could be on a local filesystem or a remote git repository +The test command provides a facility to test resources against policies by comparing expected results, declared ahead of time in a test manifest file, to actual results reported by Kyverno. Users provide the path to the folder containing a kyverno-test.yaml file where the location could be on a local filesystem or a remote git repository. ` var exampleHelp = ` +# Test a git repository containing Kyverno test cases. kyverno test https://github.com/kyverno/policies/pod-security --git-branch main - + - Executing require-non-root-groups... - applying 1 policy to 2 resources... +Executing require-non-root-groups... +applying 1 policy to 2 resources... - │───│─────────────────────────│──────────────────────────│──────────────────────────────────│────────│ - │ # │ POLICY │ RULE │ RESOURCE │ RESULT │ - │───│─────────────────────────│──────────────────────────│──────────────────────────────────│────────│ - │ 1 │ require-non-root-groups │ check-runasgroup │ default/Pod/fs-group0 │ Pass │ - │ 2 │ require-non-root-groups │ check-supplementalGroups │ default/Pod/fs-group0 │ Pass │ - │ 3 │ require-non-root-groups │ check-fsGroup │ default/Pod/fs-group0 │ Pass │ - │ 4 │ require-non-root-groups │ check-supplementalGroups │ default/Pod/supplemental-groups0 │ Pass │ - │ 5 │ require-non-root-groups │ check-fsGroup │ default/Pod/supplemental-groups0 │ Pass │ - │ 6 │ require-non-root-groups │ check-runasgroup │ default/Pod/supplemental-groups0 │ Pass │ - │───│─────────────────────────│──────────────────────────│──────────────────────────────────│────────│ - +│───│─────────────────────────│──────────────────────────│──────────────────────────────────│────────│ +│ # │ POLICY │ RULE │ RESOURCE │ RESULT │ +│───│─────────────────────────│──────────────────────────│──────────────────────────────────│────────│ +│ 1 │ require-non-root-groups │ check-runasgroup │ default/Pod/fs-group0 │ Pass │ +│ 2 │ require-non-root-groups │ check-supplementalGroups │ default/Pod/fs-group0 │ Pass │ +│ 3 │ require-non-root-groups │ check-fsGroup │ default/Pod/fs-group0 │ Pass │ +│ 4 │ require-non-root-groups │ check-supplementalGroups │ default/Pod/supplemental-groups0 │ Pass │ +│ 5 │ require-non-root-groups │ check-fsGroup │ default/Pod/supplemental-groups0 │ Pass │ +│ 6 │ require-non-root-groups │ check-runasgroup │ default/Pod/supplemental-groups0 │ Pass │ +│───│─────────────────────────│──────────────────────────│──────────────────────────────────│────────│ + + +# Test a local folder containing test cases. +kyverno test . + +Executing limit-containers-per-pod... +applying 1 policy to 4 resources... + +│───│──────────────────────────│──────────────────────────────────────│─────────────────────────────│────────│ +│ # │ POLICY │ RULE │ RESOURCE │ RESULT │ +│───│──────────────────────────│──────────────────────────────────────│─────────────────────────────│────────│ +│ 1 │ limit-containers-per-pod │ limit-containers-per-pod-bare │ default/Pod/myapp-pod-1 │ Pass │ +│ 2 │ limit-containers-per-pod │ limit-containers-per-pod-bare │ default/Pod/myapp-pod-2 │ Pass │ +│ 3 │ limit-containers-per-pod │ limit-containers-per-pod-controllers │ default/Deployment/mydeploy │ Pass │ +│ 4 │ limit-containers-per-pod │ limit-containers-per-pod-cronjob │ default/CronJob/mycronjob │ Pass │ +│───│──────────────────────────│──────────────────────────────────────│─────────────────────────────│────────│ + +Test Summary: 4 tests passed and 0 tests failed -Test file structure: +**TEST FILE STRUCTURE**: -The test.yaml has four parts: - "policies" --> List of policies which are applied. - "resources" --> List of resources on which the policies are applied. - "variables" --> Variable file path (optional). - "results" --> List of results expected after applying the policies on the resources. +The kyverno-test.yaml has four parts: + "policies" --> List of policies which are applied. + "resources" --> List of resources on which the policies are applied. + "variables" --> Variable file path containing variables referenced in the policy (OPTIONAL). + "results" --> List of results expected after applying the policies to the resources. -Test file format: +** TEST FILE FORMAT**: -For validate policies +name: +policies: +- +- +resources: +- +- +variables: (OPTIONAL) +results: +- policy: (For Namespaced [Policy] files, format is /) + rule: + resource: + namespace: (OPTIONAL) + kind: + patchedResource: (For mutate policies/rules only) + result: -- name: test-1 - policies: - - - - +**VARIABLES FILE FORMAT**: + +policies: +- name: + rules: + - name: + # Global variable values + values: + foo: bar resources: - - - - - results: - - policy: - rule: - resource: - namespace: (OPTIONAL) - kind: - result: + - name: + # Resource-specific variable values + values: + foo: baz + - name: + values: + foo: bin +**RESULT DESCRIPTIONS**: -For mutate policies - -Policy (Namespaced) - -- name: test-1 - policies: - - - - - resources: - - - - - results: - - policy: / - rule: - resource: - namespace: (OPTIONAL) - kind: - patchedResource: - result: - -ClusterPolicy (Cluster-wide) - -- name: test-1 - policies: - - - - - resources: - - - - - results: - - policy: - rule: - resource: - namespace: (OPTIONAL) - kind: - patchedResource: - result: - -Result descriptions: - -pass --> The patched resource generated by Kyverno equals the patched resource provided by the user. -fail --> The patched resource generated by Kyverno is not equal to the patched resource provided by the user. +pass --> The resource is either validated by the policy or, if a mutation, equals the state of the patched resource. +fail --> The resource fails validation or the patched resource generated by Kyverno is not equal to the input resource provided by the user. skip --> The rule is not applied. For more information visit https://kyverno.io/docs/kyverno-cli/#test @@ -138,10 +135,11 @@ For more information visit https://kyverno.io/docs/kyverno-cli/#test // Command returns version command func Command() *cobra.Command { var cmd *cobra.Command + var testFile []byte var valuesFile, fileName, gitBranch string cmd = &cobra.Command{ - Use: "test [flags]\n kyverno test --git-branch ", - Args: cobra.ExactArgs(1), + Use: "test [flags]\n kyverno test --git-branch \n kyverno test --manifest-mutate > kyverno-test.yaml\n kyverno test --manifest-validate > kyverno-test.yaml", + // Args: cobra.ExactArgs(1), Short: "run tests from directory", Long: longHelp, Example: exampleHelp, @@ -155,6 +153,48 @@ func Command() *cobra.Command { } }() + mStatus, _ := cmd.Flags().GetBool("manifest-mutate") + vStatus, _ := cmd.Flags().GetBool("manifest-validate") + if mStatus { + testFile = []byte(`name: +policies: +- +- +resources: +- +- +variables: (OPTIONAL) +results: +- policy: (For Namespaced [Policy] files, format is /) + rule: + resource: + namespace: (OPTIONAL) + kind: + patchedResource: + result: `) + fmt.Println(string(testFile)) + return nil + } + if vStatus { + testFile = []byte(`name: +policies: +- +- +resources: +- +- +variables: (OPTIONAL) +results: +- policy: (For Namespaced [Policy] files, format is /) + rule: + resource: + namespace: (OPTIONAL) + kind: + result: `) + fmt.Println(string(testFile)) + return nil + } + _, err = testCommandExecute(dirPath, valuesFile, fileName, gitBranch) if err != nil { log.Log.V(3).Info("a directory is required") @@ -166,6 +206,9 @@ func Command() *cobra.Command { } cmd.Flags().StringVarP(&fileName, "file-name", "f", "kyverno-test.yaml", "test filename") cmd.Flags().StringVarP(&gitBranch, "git-branch", "b", "", "test github repository branch") + + cmd.Flags().BoolP("manifest-mutate", "", false, "prints out a template test manifest for a mutate policy") + cmd.Flags().BoolP("manifest-validate", "", false, "prints out a template test manifest for a validate policy") return cmd }