Compliance-to-Policy (C2P) is designed to bridge Compliance as Code such as Open Security Controls Assessment Language (OSCAL) and Policy as Code used by Policy Validation Point (PVP). It generates policies in native format of PVP from OSCAL Component Definitions and produces OSCAL Assessment Results from the native assessment results of PVP. C2P can be used both as a command-line tool and a Python library, making it easy and flexible to integrate into your Continuous Compliance pipelines, such as GitHub Actions, Tekton Pipelines, or Agile Authoring Pipelines. It supports multiple PVP engines, including Kyverno, Open Cluster Management Policy Framework, and the open-source Auditree, through dedicated plugins for each. Custom plugins can be implemented with a small amount of Python code.
- Compliance-to-Policy (C2P) is running in GitOps Pipeline, Kubernetes controller, or Python/Go environment
- C2P receives Compliance as Code, for example OSCAL Component Definition that represents mapping between controls and policies (policy names/ids)
- C2P generates policies through plugin for each policy engine
- The plugin is responsible for implementing a function that takes policy names/ids and returns policies
- Policies are delivered to policy engines by GitOps sync, the subsequence pipeline task, Kubernetes controller, or a deployment automation program
- Results are collected from policy engines by a scheduled task or Kubernetes controller
- C2P aggregates the results of policy engines by controls through plugin for each policy engine
- The plugin is responsible for implementing a function that takes the results of the policy engine and returns verdicts (pass/fail/error), reason, and/or resource name for each respective policy by its names/IDs.
- C2P produces Compliance Assessment Results, for example OSCAL Assessment Results that represents the assessment results of each control
Demo:
Provide seamless integration with compliance frameworks and existing policy engines, and enable to use heterogeneous policy engines in compliance check operation
- Flexibility in choise of policy engines and compliance frameworks
- Provide plugins to cover various policy engines including proprietary/open source policy validation/enforcement engines, or in-house policy validation/enforcement program
- Cover various compliance frameworks not only OSCAL but also other GRC frameworks and Cloud Security Posture Management services
- Community-driven plugin extension
- Provide an efficient plugin interface and development method
- Open Security Controls Assessment Language (OSCAL)
- OSCAL standard provides a compliance framework and the corresponding set of key compliance artifacts expressed in machine processable formats enabling all compliance documents to be treated as code and therefore processed and managed in the same manner.
- Kyverno (for Kubernetes resources)
- Kyverno is a policy engine designed for Kubernetes, where policies are managed as Kubernetes resources. Kyverno policies can validate, mutate, generate, and clean up Kubernetes resources.
- Open Cluster Management Policy Framework (for Kubernetes resources)
- OCM is a multi-cluster management platform that provides governance of Kubernetes policies. Its policy framework allows for the validation and enforcement of policies across multiple clusters.
- Auditree (for any target, especially well-suited for resources of PaaS/SaaS/IaaS available through REST API.)
- Auditree is a GitOps based workflow automation that enables the collection and verification of evidence, building a long-term store of evidence in an git "evidence locker." Evidence is gathered by code scripts called "fetchers" and verified by "checks."
Roadmap:
- OPA/Gatekeeper (for Kubernetes resources)
- Ansible (for any target, especially for VMs and OnPremise hosts)
The Go verion is available in the go directory.
pip install git+/~https://github.com/oscal-compass/compliance-to-policy.git
You may be asked passphrase of SSH key to access to the git repo.
- Clone the repository
git clone /~https://github.com/oscal-compass/compliance-to-policy.git
- Go to
compliance-to-policy
cd compliance-to-policy
- Install
make install
- Generate Kyverno Policy (C2P Compliance to Policy)
E.g.
python samples_public/kyverno/compliance_to_policy.py -o /tmp/deliverable-policy
$ python samples_public/kyverno/compliance_to_policy.py -o /tmp/deliverable-policy tree /tmp/deliverable-policy disallow-capabilities - disallow-capabilities.yaml allowed-base-images - 02-setup-cm.yaml - allowed-base-images.yaml
- Deploy the generated policies
E.g.
kubectl apply -R -f /tmp/deliverable-policy
$ kubectl apply -R -f /tmp/deliverable-policy namespace/platform created configmap/baseimages created Warning: Validation failure actions enforce/audit are deprecated, use Enforce/Audit instead. clusterpolicy.kyverno.io/allowed-base-images created clusterpolicy.kyverno.io/disallow-capabilities created
- Check policy results
$ kubectl get policyreport,clusterpolicyreport -A NAMESPACE NAME PASS FAIL WARN ERROR SKIP AGE kube-system policyreport.wgpolicyk8s.io/cpol-allowed-base-images 0 12 0 0 0 19s kube-system policyreport.wgpolicyk8s.io/cpol-disallow-capabilities 9 2 0 0 0 19s kyverno policyreport.wgpolicyk8s.io/cpol-allowed-base-images 0 18 0 0 0 9s kyverno policyreport.wgpolicyk8s.io/cpol-disallow-capabilities 18 0 0 0 0 9s local-path-storage policyreport.wgpolicyk8s.io/cpol-allowed-base-images 0 3 0 0 0 16s local-path-storage policyreport.wgpolicyk8s.io/cpol-disallow-capabilities 3 0 0 0 0 16s
- Collect policy/cluster policy reports as PVP Raw results
kubectl get policyreport -A -o yaml > /tmp/policyreports.wgpolicyk8s.io.yaml kubectl get clusterpolicyreport -o yaml > /tmp/clusterpolicyreports.wgpolicyk8s.io.yaml
- Generate Assessment Result (C2P Result to Compliance)
python samples_public/kyverno/result_to_compliance.py \ -polr /tmp/policyreports.wgpolicyk8s.io.yaml \ -cpolr /tmp/clusterpolicyreports.wgpolicyk8s.io.yaml \ > /tmp/assessment_results.json
- OSCAL Assessment Results is not human readable format. You can see the merged report in markdown by a quick viewer.
c2p tools viewer -ar /tmp/assessment_results.json -cdef ./plugins_public/tests/data/kyverno/component-definition.json -o /tmp/assessment_results.md
- Kyverno
- Open Cluster Management Governance Policy Framework
- Auditree
- Heterogeneous PVPs (mixing Kyverno, OCM Policy, and Auditree)
- Create
C2PConfig
object to supply compliance requirements and some metadata (See also kyverno/compliance_to_policy.py for a real example)c2p_config = C2PConfig() c2p_config.compliance = ComplianceOscal() c2p_config.compliance.component_definition = 'plugins_public/tests/data/kyverno/component-definition.json' c2p_config.pvp_name = 'Kyverno' c2p_config.result_title = 'Kyverno Assessment Results' c2p_config.result_description = 'OSCAL Assessment Results from Kyverno'
- Select a plugin for supported PVPs (
PluginKyverno
,PluginOCM
) and createPluginConfig
object to supply plugin specific propertiesfrom plugins_public.plugins.kyverno import PluginConfigKyverno, PluginKyverno policy_template_dir = 'plugins_public/tests/data/kyverno/policy-resources' config = PluginConfigKyverno(policy_template_dir=policy_template_dir, deliverable_policy_dir='/tmp/deliverable-policies')
- Create
C2P
andPlugin
c2p = C2P(c2p_config) plugin = PluginKyverno(config)
- Get policy from
c2p
and generate PVP policy bygenerate_pvp_policy()
policy = c2p.get_policy() plugin.generate_pvp_policy(policy)
- The deliverable policies are output in '/tmp/deliverable-policies'
$ tree /tmp/deliverable-policy /tmp/deliverable-policy ├── allowed-base-images │ ├── 02-setup-cm.yaml │ └── allowed-base-images.yaml └── disallow-capabilities └── disallow-capabilities.yaml
- Create
C2PConfig
object to supply compliance requirements and some metadata (See also kyverno/compliance_to_policy.py for a real example)c2p_config = C2PConfig() c2p_config.compliance = ComplianceOscal() c2p_config.compliance.component_definition = 'plugins_public/tests/data/kyverno/component-definition.json' c2p_config.pvp_name = 'Kyverno' c2p_config.result_title = 'Kyverno Assessment Results' c2p_config.result_description = 'OSCAL Assessment Results from Kyverno'
- Select a plugin for supported PVPs (
PluginKyverno
,PluginOCM
) and createPluginConfig
object to supply plugin specific propertiesfrom plugins_public.plugins.kyverno import PluginConfigKyverno, PluginKyverno config = PluginConfigKyverno()
- Create
C2P
andPlugin
c2p = C2P(c2p_config) plugin = PluginKyverno(config)
- Load PVP native results
policy_report_file = 'plugins_public/tests/data/kyverno/policyreports.wgpolicyk8s.io.yaml' cluster_policy_report_file = 'plugins_public/tests/data/kyverno/clusterpolicyreports.wgpolicyk8s.io.yaml' policy_report = yaml.safe_load(pathlib.Path(policy_report_file).open('r')) cluster_policy = yaml.safe_load(pathlib.Path(cluster_policy_report_file).open('r')) pvp_raw_result = RawResult(data=policy_report['items'] + cluster_policy['items'])
- Call
generate_pvp_result()
of the plugin to get a formatted PVP resultpvp_result = PluginKyverno().generate_pvp_result(pvp_raw_result)
- Create
C2P
and callresult_to_oscal()
to obtain Compliance Assessment Resultsc2p.set_pvp_result(pvp_result) oscal_assessment_results = c2p.result_to_oscal() print(oscal_assessment_results.oscal_serialize_json(pretty=True))
- (Optional) you may reformat OSCAL Assessment Results in markdown style.
c2p tools viewer -ar <OSCAL Assessment Results (json)> -cdef ./plugins_public/tests/data/ocm/component-definition.json -o /tmp/assessment_results.md
You can create a custom plugin by overriding PluginSpec
and PluginConfig
.
PluginSpec
has two interfaces generate_pvp_policy
and generate_pvp_result
.
C2P framework will instantiate PluginSpec
with PluginConfig
.
- Extend PluginConfig with custom fields as the plugin needs
from c2p.framework.plugin_spec import PluginSpec class YourPluginConfig(PluginConfig): custom_field: str = Field(..., title='Custom field for your plugin')
- Extend PluginSpec and define init with YourPluginConfig
class YourPlugin(PluginSpec): def __init__(self, config: Optional[YourPluginConfig] = None) -> None: super().__init__() self.config = config # work on config
generate_pvp_policy()
inPluginSpec
accepts one argumentpolicy: c2p.framework.models.Policy
. The object has two fields (rule_sets
andparameters
).rule_sets
andparameters
are a list of Rule_Id, Check_Id, Parameter_Id, Parameter_Value, etc of the components handled by your PVP in OSCAL Component Definition.- Implement the logic to generate PVP policy from provided rule_sets and parameters.
def generate_pvp_policy(self, policy: Policy): rule_sets: List[RuleSet] = policy.rule_sets parameters: List[Parameter] = policy.parameters # generate deliverable policy from rule_sets and parameters
-
generate_pvp_result()
is expected to generate the summarized raw results of your PVP per unit inPVPResult
format. This unit must be associated with a unique id called Check_Id. For example of PluginKyverno, Policy Reports is the raw results and are summarized by policy name.def generate_pvp_result(self, raw_result: RawResult) -> PVPResult: pvp_result: PVPResult = PVPResult() observations: List[ObservationByCheck] = [] polrs = list( filter( lambda x: x['apiVersion'] == 'wgpolicyk8s.io/v1alpha2' and x['kind'] == 'PolicyReport', raw_result.data ) ) cpolrs = list( filter( lambda x: x['apiVersion'] == 'wgpolicyk8s.io/v1alpha2' and x['kind'] == 'ClusterPolicyReport', raw_result.data, ) ) results = [] for polr in polrs: for result in polr['results']: results.append(result) for cpolr in cpolrs: for result in cpolr['results']: results.append(result) policy_names = list(map(lambda x: x['policy'], results)) # policy_name is used as check_id policy_names = set(policy_names) for policy_name in policy_names: observation = ObservationByCheck(check_id=policy_name, methods=['AUTOMATED'], collected=get_datetime())
-
The input argument
raw_result
hasdata
field that is serialized raw results as dict. You can define your preferable format of the data. C2P Framework will pass PVP native results to plugin with this format.
- Put the plugin in plugin directory /plugins_public/plugins or Python module path when you use C2P.
- Install Python
$ python --version Python 3.10.12
- Setup venv
python -m venv .venv
- Install dependant modules
make install-dev
- Enable detect-secret
pre-commit install
$ make test
plugins_public/tests/plugins/test_kyverno.py::test_kyverno_pvp_result_to_compliance PASSED [ 25%]
plugins_public/tests/plugins/test_kyverno.py::test_kyverno_compliance_to_policy PASSED [ 50%]
plugins_public/tests/plugins/test_ocm.py::test_ocm_pvp_result_to_compliance PASSED [ 75%]
plugins_public/tests/plugins/test_ocm.py::test_ocm_compliance_to_policy
------------------------------------------------------------------------------------------ live log call -------------------------------------------------------------------------------------------
2024-04-25 05:31:48 [ INFO] The deliverable policy directory '/var/folders/yx/1mv5rdh53xd93bphsc459ht00000gn/T/tmpxtvpcrpr/deliverable-policy' is not found. Creating... (ocm.py:191)
PASSED [100%]
======================================================================================== 4 passed in 0.31s =========================================================================================
tests/c2p/framework/test_c2p.py::test_result_to_oscal PASSED [ 33%]
tests/c2p/test_cli.py::test_run PASSED [ 66%]
tests/c2p/test_cli.py::test_version PASSED [100%]
======================================================================================== 3 passed in 0.26s =========================================================================================
make clean
If you would like to see the detailed LICENSE click here. Consult contributors for a list of authors and maintainers for the core team.
# Copyright (c) 2024 The OSCAL Compass Authors. All rights reserved.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# https://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
We are a Cloud Native Computing Foundation sandbox project.
The Linux Foundation® (TLF) has registered trademarks and uses trademarks. For a list of TLF trademarks, see Trademark Usage".
Compliance to Policy was originally created by IBM.