- Overview
- Requirements
- Running the Scorecard
- Configuration
- Tests Performed
- Exit Status
- Extending the Scorecard with Plugins
- Running the scorecard with an OLM-managed operator
The scorecard works by creating all resources required by CRs and the operator.
The scorecard will create another container in the operator deployment which is used to record calls to the API server and run a lot of the tests. The tests performed will also examine some of the fields in the CRs.
The scorecard also supports plugins which allows to extend the functionality of the scorecard and add additional tests on it.
Following are some requirements for the operator project which would be checked by the scorecard.
- Access to a Kubernetes v1.11.3+ cluster
For non-SDK operators:
- Resource manifests for installing/configuring the operator and custom resources. (see the Writing E2E Tests doc for more information on the global and namespaced manifests)
- Config getter that supports reading from the
KUBECONFIG
environment variable (such as theclientcmd
orcontroller-runtime
config getters). This is required for the scorecard proxy to work correctly.
NOTE: If you would like to use it to check the integration of your operator project with OLM then also the Cluster Service Version (CSV) file will be required. This is a requirement when the olm-deployed
option is used
- Setup the
.osdk-scorecard.yaml
configuration file in your project. See Config file - Create the namespace defined in the RBAC files(
role_binding
) - Then, run the scorecard, for example
$ operator-sdk scorecard
. See the Command args to check its options.
NOTE: If your operator is non-SDK then some steps will be required in order to meet its requirements.
The scorecard is configured by a config file that allows configuring internal plugins as well as a few global configuration options.
To use scorecard, you need to create a config file which by default will be <project_dir>/.osdk-scorecard.yaml
.The following is an example of how the config file may look:
scorecard:
# Setting a global scorecard option
output: json
plugins:
# `basic` tests configured to test 2 CRs
- basic:
cr-manifest:
- "deploy/crds/cache.example.com_v1alpha1_memcached_cr.yaml"
- "deploy/crds/cache.example.com_v1alpha1_memcachedrs_cr.yaml"
# `olm` tests configured to test 2 CRs
- olm:
cr-manifest:
- "deploy/crds/cache.example.com_v1alpha1_memcached_cr.yaml"
- "deploy/crds/cache.example.com_v1alpha1_memcachedrs_cr.yaml"
csv-path: "deploy/olm-catalog/memcached-operator/0.0.3/memcached-operator.v0.0.3.clusterserviceversion.yaml"
The hierarchy of config methods for the global options that are also configurable via a flag from highest priority to least is: flag->file->default.
The config file support is provided by the viper
package. For more info on how viper configuration works, see viper
's README.
NOTE: The config file can be in any of the json
, yaml
, or toml
formats as long as the file has the correct extension. As the config file may be extended to allow configuration
of all operator-sdk
subcommands in the future, the scorecard's configuration must be under a scorecard
subsection.
While most configuration is done via a config file, there are a few important args that can be used as follows.
Flag | Type | Description |
---|---|---|
--bundle , -b |
string | The path to a bundle directory used for the bundle validation test. |
--config |
string | Path to config file (default <project_dir>/.osdk-scorecard.yaml ; file type and extension must be .yaml ). If a config file is not provided and a config file is not found at the default location, the scorecard will exit with an error. |
--output , -o |
string | Output format. Valid options are: text and json . The default format is text , which is designed to be a simpler human readable format. The json format uses the JSON schema output format used for plugins defined later in this document. |
--kubeconfig , -o |
string | path to kubeconfig. It sets the kubeconfig internally for internal plugins. |
--version |
string | The version of scorecard to run, v1alpha2 is the default, valid values are v1alpha2. |
--selector , -l |
string | The label selector to filter tests on. |
--list , -L |
bool | If true, only print the test names that would be run based on selector filtering. |
Option | Type | Description |
---|---|---|
bundle |
string | equivalent of the --bundle flag. OLM bundle directory path, when specified runs bundle validation |
output |
string | equivalent of the --output flag. If this option is defined by both the config file and the flag, the flag's value takes priority |
kubeconfig |
string | equivalent of the --kubeconfig flag. If this option is defined by both the config file and the flag, the flag's value takes priority |
plugins |
array | this is an array of Plugins. |
A plugin object is used to configure plugins. The possible values for the plugin object are basic
, or olm
.
Note that each Plugin type has different configuration options and they are named differently in the config. Only one of these fields can be set per plugin.
The basic
and olm
internal plugins have the same configuration fields:
Option | Type | Description |
---|---|---|
cr-manifest |
[]string | path(s) for CRs being tested.(required if olm-deployed is not set or false) |
csv-path |
string | path to CSV for the operator (required for OLM tests or if olm-deployed is set to true) |
olm-deployed |
bool | indicates that the CSV and relevant CRD's have been deployed onto the cluster by the Operator Lifecycle Manager (OLM) |
kubeconfig |
string | path to kubeconfig. If both the global kubeconfig and this field are set, this field is used for the plugin |
namespace |
string | namespace to run the plugins in. If not set, the default specified by the kubeconfig is used |
init-timeout |
int | time in seconds until a timeout during initialization or cleanup of the operator |
crds-dir |
string | path to directory containing CRDs that must be deployed to the cluster |
namespaced-manifest |
string | manifest file with all resources that run within a namespace. By default, the scorecard will combine service_account.yaml , role.yaml , role_binding.yaml , and operator.yaml from the deploy directory into a temporary manifest to use as the namespaced manifest |
global-manifest |
string | manifest containing required resources that run globally (not namespaced). By default, the scorecard will combine all CRDs in the crds-dir directory into a temporary manifest to use as the global manifest |
proxy-port |
int | port for scorecard-proxy to listen to, default is port 8889 |
Following the description of each internal Plugin. Note that are 8 internal tests across 2 internal plugins that the scorecard can run. If multiple CRs are specified for a plugin, the test environment is fully cleaned up after each CR so each CR gets a clean testing environment.
Each test has a short name
that uniquely identifies the test. This is useful for selecting a specific test or tests to run as follows:
operator-sdk scorecard -o text --selector=test=checkspectest
operator-sdk scorecard -o text --selector='test in (checkspectest,checkstatustest)'
Test | Description | Short Name |
---|---|---|
Spec Block Exists | This test checks the Custom Resource(s) created in the cluster to make sure that all CRs have a spec block. This test has a maximum score of 1 | checkspectest |
Status Block Exists | This test checks the Custom Resource(s) created in the cluster to make sure that all CRs have a status block. This test has a maximum score of 1 | checkstatustest |
Writing Into CRs Has An Effect | This test reads the scorecard proxy's logs to verify that the operator is making PUT and/or POST requests to the API server, indicating that it is modifying resources. This test has a maximum score of 1 |
writingintocrshaseffecttest |
Test | Description | Short Name |
---|---|---|
OLM Bundle Validation | This test validates the OLM bundle manifests found in the bundle directory as specifed by the bundle flag. If the bundle contents contain errors, then the test result output will include the validator log as well as error messages from the validation library. See this document for details on OLM bundles. | bundlevalidationtest |
Provided APIs have validation | This test verifies that the CRDs for the provided CRs contain a validation section and that there is validation for each spec and status field detected in the CR. This test has a maximum score equal to the number of CRs provided via the cr-manifest option. |
crdshavevalidationtest |
Owned CRDs Have Resources Listed | This test makes sure that the CRDs for each CR provided via the cr-manifest option have a resources subsection in the owned CRDs section of the CSV. If the test detects used resources that are not listed in the resources section, it will list them in the suggestions at the end of the test. This test has a maximum score equal to the number of CRs provided via the cr-manifest option. |
crdshaveresourcestest |
Spec Fields With Descriptors | This test verifies that every field in the Custom Resources' spec sections have a corresponding descriptor listed in the CSV. This test has a maximum score equal to the total number of fields in the spec sections of each custom resource passed in via the cr-manifest option. |
specdescriptorstest |
Status Fields With Descriptors | This test verifies that every field in the Custom Resources' status sections have a corresponding descriptor listed in the CSV. This test has a maximum score equal to the total number of fields in the status sections of each custom resource passed in via the cr-manifest option. |
statusdescriptorstest |
The scorecard return code is 1 if any of the tests executed did not pass and 0 if all selected tests pass.
To allow the scorecard to be further extended and capable of more complex testing as well as allow the community to make their own scorecard tests, a plugin system has been implemented for the scorecard. To use it, a plugin developer simply needs to provide the binary or script, and the user can then configure the scorecard to use the new plugin. Since the scorecard can run any executable as a plugin, the plugins can be written in any programming language supported by the OS the scorecard is being run on. All plugins are run from the root of the operator project.
To provide results to the scorecard, the plugin must output a valid JSON object to its stdout
. Invalid JSON in stdout
will result in the plugin being marked as errored.
To provide logs to the scorecard, plugins can either set the log
field for the scorecard suites they return or they can output logs to stderr
, which will stream the log
to the console if the scorecard is being run in with output
unset or set to text
, or be added to the main ScorecardOutput.Log
field when output
is set to json
The JSON output is formatted in the same way that a Kubernetes API would be, which allows for updates to the schema as well as the use of various Kubernetes helpers. The Golang structs are defined in pkg/apis/scorecard/v1alpha2/types.go
and can be easily implemented by plugins written in Golang. Below is the JSON Schema:
{
"$schema": "http://json-schema.org/draft-04/schema#",
"$ref": "#/definitions/ScorecardOutput",
"definitions": {
"FieldsV1": {
"additionalProperties": false,
"type": "object"
},
"ManagedFieldsEntry": {
"properties": {
"apiVersion": {
"type": "string"
},
"fieldsType": {
"type": "string"
},
"fieldsV1": {
"$schema": "http://json-schema.org/draft-04/schema#",
"$ref": "#/definitions/FieldsV1"
},
"manager": {
"type": "string"
},
"operation": {
"type": "string"
},
"time": {
"$ref": "#/definitions/Time"
}
},
"additionalProperties": false,
"type": "object"
},
"ObjectMeta": {
"properties": {
"annotations": {
"patternProperties": {
".*": {
"type": "string"
}
},
"type": "object"
},
"clusterName": {
"type": "string"
},
"creationTimestamp": {
"$schema": "http://json-schema.org/draft-04/schema#",
"$ref": "#/definitions/Time"
},
"deletionGracePeriodSeconds": {
"type": "integer"
},
"deletionTimestamp": {
"$ref": "#/definitions/Time"
},
"finalizers": {
"items": {
"type": "string"
},
"type": "array"
},
"generateName": {
"type": "string"
},
"generation": {
"type": "integer"
},
"labels": {
"patternProperties": {
".*": {
"type": "string"
}
},
"type": "object"
},
"managedFields": {
"items": {
"$schema": "http://json-schema.org/draft-04/schema#",
"$ref": "#/definitions/ManagedFieldsEntry"
},
"type": "array"
},
"name": {
"type": "string"
},
"namespace": {
"type": "string"
},
"ownerReferences": {
"items": {
"$schema": "http://json-schema.org/draft-04/schema#",
"$ref": "#/definitions/OwnerReference"
},
"type": "array"
},
"resourceVersion": {
"type": "string"
},
"selfLink": {
"type": "string"
},
"uid": {
"type": "string"
}
},
"additionalProperties": false,
"type": "object"
},
"OwnerReference": {
"required": [
"apiVersion",
"kind",
"name",
"uid"
],
"properties": {
"apiVersion": {
"type": "string"
},
"blockOwnerDeletion": {
"type": "boolean"
},
"controller": {
"type": "boolean"
},
"kind": {
"type": "string"
},
"name": {
"type": "string"
},
"uid": {
"type": "string"
}
},
"additionalProperties": false,
"type": "object"
},
"ScorecardOutput": {
"required": [
"TypeMeta",
"log",
"results"
],
"properties": {
"TypeMeta": {
"$schema": "http://json-schema.org/draft-04/schema#",
"$ref": "#/definitions/TypeMeta"
},
"log": {
"type": "string"
},
"metadata": {
"$schema": "http://json-schema.org/draft-04/schema#",
"$ref": "#/definitions/ObjectMeta"
},
"results": {
"items": {
"$schema": "http://json-schema.org/draft-04/schema#",
"$ref": "#/definitions/ScorecardTestResult"
},
"type": "array"
}
},
"additionalProperties": false,
"type": "object"
},
"ScorecardTestResult": {
"required": [
"name",
"description"
],
"properties": {
"description": {
"type": "string"
},
"errors": {
"items": {
"type": "string"
},
"type": "array"
},
"labels": {
"patternProperties": {
".*": {
"type": "string"
}
},
"type": "object"
},
"log": {
"type": "string"
},
"name": {
"type": "string"
},
"state": {
"type": "string"
},
"suggestions": {
"items": {
"type": "string"
},
"type": "array"
}
},
"additionalProperties": false,
"type": "object"
},
"Time": {
"additionalProperties": false,
"type": "object"
},
"TypeMeta": {
"properties": {
"apiVersion": {
"type": "string"
},
"kind": {
"type": "string"
}
},
"additionalProperties": false,
"type": "object"
}
}
}
NOTE: The ScorecardOutput
object is designed the same as a Kubernetes API, and thus also has a full TypeMeta
and ObjectMeta
. This means that it contains
various other fields such as selfLink
, uid
, and others. At the moment, the only required fields and the only fields that will be checked by the scorecard
are the kind
and apiVersion
fields as listed in the above JSONSchema.
Example of a valid JSON output:
{
"kind": "ScorecardOutput",
"apiVersion": "osdk.openshift.io/v1alpha2",
"metadata": {
"creationTimestamp": null
},
"log": "time=\"2020-01-16T15:30:41-06:00\" level=info msg=\"Using config file: /home/someuser/projects/memcached-operator/.osdk-scorecard.yaml\"\n",
"results": [
{
"name": "Spec Block Exists",
"description": "Custom Resource has a Spec Block",
"labels": {
"necessity": "required",
"suite": "basic",
"test": "checkspectest"
},
"state": "pass"
},
{
"name": "Status Block Exists",
"description": "Custom Resource has a Status Block",
"labels": {
"necessity": "required",
"suite": "basic",
"test": "checkstatustest"
},
"state": "pass"
},
{
"name": "Writing into CRs has an effect",
"description": "A CR sends PUT/POST requests to the API server to modify resources in response to spec block changes",
"labels": {
"necessity": "required",
"suite": "basic",
"test": "writingintocrshaseffecttest"
},
"state": "pass"
},
{
"name": "Bundle Validation Test",
"description": "Validates bundle contents",
"labels": {
"necessity": "required",
"suite": "olm",
"test": "bundlevalidationtest"
},
"state": "fail",
"errors": [
"unable to find the OLM 'bundle' directory which is required for this test"
]
},
{
"name": "Provided APIs have validation",
"description": "All CRDs have an OpenAPI validation subsection",
"labels": {
"necessity": "required",
"suite": "olm",
"test": "crdshavevalidationtest"
},
"state": "pass"
},
{
"name": "Owned CRDs have resources listed",
"description": "All Owned CRDs contain a resources subsection",
"labels": {
"necessity": "required",
"suite": "olm",
"test": "crdshaveresourcestest"
},
"state": "fail",
"suggestions": [
"If it would be helpful to an end-user to understand or troubleshoot your CR, consider adding resources [memcacheds/v1alpha1 replicasets/v1 deployments/v1 services/v1 servicemonitors/v1 pods/v1 configmaps/v1] to the resources section for owned CRD Memcached"
]
},
{
"name": "Spec fields with descriptors",
"description": "All spec fields have matching descriptors in the CSV",
"labels": {
"necessity": "required",
"suite": "olm",
"test": "specdescriptorstest"
},
"state": "fail",
"suggestions": [
"Add a spec descriptor for size"
]
},
{
"name": "Status fields with descriptors",
"description": "All status fields have matching descriptors in the CSV",
"labels": {
"necessity": "required",
"suite": "olm",
"test": "statusdescriptorstest"
},
"state": "fail",
"suggestions": [
"Add a status descriptor for status"
]
}
]
}
NOTE: The ScorecardOutput.Log
field is only intended to be used to log the scorecard's output and the scorecard will ignore that field if a plugin provides it.
To add logs to the main ScorecardOuput.Log
field, a plugin can output the logs to stderr
.
The scorecard can be run using a Cluster Service Version (CSV), providing a way to test cluster-ready and non-SDK operators.
Running with a CSV alone requires both the csv-path: <CSV manifest path>
and olm-deployed
options to be set. The scorecard assumes your CSV and relevant CRD's have been deployed onto the cluster using OLM when using olm-deployed
.
The scorecard requires a proxy container in the operator's Deployment
pod to read operator logs. A few modifications to your CSV and creation of one extra object are required to run the proxy before deploying your operator with OLM:
- Create a proxy server secret containing a local Kubeconfig:
- Generate a username using the scorecard proxy's namespaced owner reference.
# Substitute "$your_namespace" for the namespace your operator will be deployed in (if any). $ echo '{"apiVersion":"","kind":"","name":"scorecard","uid":"","Namespace":"'${your_namespace}'"}' | base64 -w 0 eyJhcGlWZXJzaW9uIjoiIiwia2luZCI6IiIsIm5hbWUiOiJzY29yZWNhcmQiLCJ1aWQiOiIiLCJOYW1lc3BhY2UiOiJvbG0ifQo=
- Write a
Config
manifestscorecard-config.yaml
using the following template, substituting${your_username}
for the base64 username generated above:apiVersion: v1 kind: Config clusters: - cluster: insecure-skip-tls-verify: true server: http://${your_username}@localhost:8889 name: proxy-server contexts: - context: cluster: proxy-server user: admin/proxy-server name: $namespace/proxy-server current-context: $namespace/proxy-server preferences: {} users: - name: admin/proxy-server user: username: ${your_username} password: unused
- Encode the
Config
as base64:$ cat scorecard-config.yaml | base64 -w 0 YXBpVmVyc2lvbjogdjEKa2luZDogQ29uZmlnCmNsdXN0ZXJzOgotIGNsdXN0ZXI6CiAgICBpbnNlY3VyZS1za2lwLXRscy12ZXJpZnk6IHRydWUKICAgIHNlcnZlcjogaHR0cDovL2V5SmhjR2xXWlhKemFXOXVJam9pSWl3aWEybHVaQ0k2SWlJc0ltNWhiV1VpT2lKelkyOXlaV05oY21RaUxDSjFhV1FpT2lJaUxDSk9ZVzFsYzNCaFkyVWlPaUp2YkcwaWZRbz1AbG9jYWxob3N0Ojg4ODkKICBuYW1lOiBwcm94eS1zZXJ2ZXIKY29udGV4dHM6Ci0gY29udGV4dDoKICAgIGNsdXN0ZXI6IHByb3h5LXNlcnZlcgogICAgdXNlcjogYWRtaW4vcHJveHktc2VydmVyCiAgbmFtZTogL3Byb3h5LXNlcnZlcgpjdXJyZW50LWNvbnRleHQ6IC9wcm94eS1zZXJ2ZXIKcHJlZmVyZW5jZXM6IHt9CnVzZXJzOgotIG5hbWU6IGFkbWluL3Byb3h5LXNlcnZlcgogIHVzZXI6CiAgICB1c2VybmFtZTogZXlKaGNHbFdaWEp6YVc5dUlqb2lJaXdpYTJsdVpDSTZJaUlzSW01aGJXVWlPaUp6WTI5eVpXTmhjbVFpTENKMWFXUWlPaUlpTENKT1lXMWxjM0JoWTJVaU9pSnZiRzBpZlFvPQogICAgcGFzc3dvcmQ6IHVudXNlZAo=
- Create a
Secret
manifestscorecard-secret.yaml
containing the operator's namespace (if any) theConfig
's base64 encoding as aspec.data
value under the keykubeconfig
:apiVersion: v1 kind: Secret metadata: name: scorecard-kubeconfig namespace: ${your_namespace} data: kubeconfig: ${kubeconfig_base64}
- Apply the secret in-cluster:
$ kubectl apply -f scorecard-secret.yaml
- Insert a volume referring to the
Secret
into the operator'sDeployment
:spec: install: spec: deployments: - name: memcached-operator spec: ... template: ... spec: containers: ... volumes: # scorecard kubeconfig volume - name: scorecard-kubeconfig secret: secretName: scorecard-kubeconfig items: - key: kubeconfig path: config
- Generate a username using the scorecard proxy's namespaced owner reference.
- Insert a volume mount and
KUBECONFIG
environment variable into each container in your operator'sDeployment
:spec: install: spec: deployments: - name: memcached-operator spec: ... template: ... spec: containers: - name: container1 ... volumeMounts: # scorecard kubeconfig volume mount - name: scorecard-kubeconfig mountPath: /scorecard-secret env: # scorecard kubeconfig env - name: KUBECONFIG value: /scorecard-secret/config - name: container2 # Do the same for this and all other containers. ...
- Insert the scorecard proxy container into the operator's
Deployment
:spec: install: spec: deployments: - name: memcached-operator spec: ... template: ... spec: containers: ... # scorecard proxy container - name: scorecard-proxy command: - scorecard-proxy env: - name: WATCH_NAMESPACE valueFrom: fieldRef: apiVersion: v1 fieldPath: metadata.namespace image: quay.io/operator-framework/scorecard-proxy:master imagePullPolicy: Always ports: - name: proxy containerPort: 8889
Alternatively, the community-operators repo has several bash functions that can perform these operations for you:
$ curl -Lo csv-manifest-modifiers.sh https://raw.githubusercontent.com/operator-framework/community-operators/master/scripts/lib/file
$ . ./csv-manifest-modifiers.sh
# $NAMESPACE is the namespace your operator will deploy in
$ create_kubeconfig_secret_file scorecard-secret.yaml "$NAMESPACE"
$ kubectl apply -f scorecard-secret.yaml
# $CSV_FILE is the path to your operator's CSV manifest
$ insert_kubeconfig_volume "$CSV_FILE"
$ insert_kubeconfig_secret_mount "$CSV_FILE"
$ insert_proxy_container "$CSV_FILE" "quay.io/operator-framework/scorecard-proxy:master"
Once done, follow the steps in this document to bundle your CSV and CRD's, deploy OLM on minikube or OKD, and deploy your operator. Once these steps have been completed, run the scorecard with both the csv-path: <CSV manifest path>
and olm-deployed
options set.
NOTES:
- As of now, using the scorecard with a CSV does not permit multiple CR manifests to be set through the CLI/config/CSV annotations. You will have to tear down your operator in the cluster, re-deploy, and re-run the scorecard for each CR being tested. In the future the scorecard will fully support testing multiple CR's without requiring users to teardown/standup each time.
- You can either set
cr-manifest
or your CSV'smetadata.annotations['alm-examples']
to provide CR's to the scorecard, but not both.