This commit is contained in:
2026-04-18 10:43:01 +07:00
parent 5ee121d49a
commit 9342806c3f
5 changed files with 142 additions and 1 deletions
@@ -28,6 +28,14 @@ spec:
configMap: configMap:
name: kyverno-global-config name: kyverno-global-config
namespace: kyverno namespace: kyverno
# - name: globalConfig
# apiCall:
# urlPath: "/api/v1/namespaces/kyverno/configmaps/kyverno-global-config"
# jmesPath: "data"
# - name: apiCredentials
# apiCall:
# urlPath: "/api/v1/namespaces/kyverno/secrets/policy-api-credentials"
# jmesPath: "data.api-key | base64_decode(@)"
mutate: mutate:
foreach: foreach:
- list: "request.object.spec.containers" - list: "request.object.spec.containers"
@@ -0,0 +1,57 @@
# Другие варианты проверок
#
# проверить, что PVC использует StorageClass из одобренного списка:
#
rules:
- name: check-storage-class
match:
resources:
kinds:
- PersistentVolumeClaim
context:
- name: storageClassInfo
apiCall:
urlPath: "/apis/storage.k8s.io/v1/storageclasses/{{ request.object.spec.storageClassName }}"
jmesPath: "metadata.labels.\"approved-for-production\""
validate:
message: >-
StorageClass '{{ request.object.spec.storageClassName }}' не одобрена для production.
Используйте StorageClass с лейблом approved-for-production: "true"
deny:
conditions:
- key: "{{ storageClassInfo }}"
operator: NotEquals
value: "true"
# проверить, что количество реплик не превышает кворум с учётом текущей нагрузки:
#
context:
- name: existingDeployments
apiCall:
urlPath: >-
/apis/apps/v1/namespaces/{{ request.object.metadata.namespace }}/deployments
jmesPath: "items[?metadata.name != '{{ request.object.metadata.name }}'] | length(@)"
validate:
message: >-
В namespace уже {{ existingDeployments }} деплойментов.
Максимум разрешено 20.
deny:
conditions:
- key: "{{ existingDeployments }}"
operator: GreaterThanOrEquals
value: "20"
# принимать решения на основе состояния нод
#
context:
- name: nodesInfo
apiCall:
urlPath: "/api/v1/nodes"
jmesPath: "items[?metadata.labels.\"node-type\" == 'gpu'].metadata.name"
validate:
message: "GPU workloads требуют минимум 2 GPU-ноды в кластере"
deny:
conditions:
- key: "{{ length(nodesInfo) }}"
operator: LessThan
value: "2"
@@ -49,4 +49,4 @@ spec:
operator: AllNotIn operator: AllNotIn
value: value:
- emergency-deployers - emergency-deployers
- system:masters - system:masters
@@ -0,0 +1,33 @@
# Полезный паттерн — разные правила для CREATE и UPDATE
rules:
- name: validate-on-create-only
match:
resources:
kinds:
- Deployment
preconditions:
any:
- key: "{{ request.operation }}"
operator: Equals
value: CREATE
validate:
# применяется только при создании
- name: validate-image-on-update
match:
resources:
kinds:
- Deployment
preconditions:
all:
- key: "{{ request.operation }}"
operator: Equals
value: UPDATE
- key: >-
{{ request.object.spec.template.spec.containers[0].image }}
operator: NotEquals
value: >-
{{ request.oldObject.spec.template.spec.containers[0].image }}
validate:
# применяется только при UPDATE с изменением образа
@@ -0,0 +1,43 @@
# .github/workflows/policy-ci.yaml
name: Kyverno Policy CI
on:
pull_request:
paths:
- 'policies/**'
jobs:
test:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v3
- name: Install Kyverno CLI
run: |
curl -LO https://github.com/kyverno/kyverno/releases/latest/download/kyverno-cli_linux_amd64.tar.gz
tar -xzf kyverno-cli_linux_amd64.tar.gz
sudo mv kyverno /usr/local/bin/
- name: Lint policies
run: |
kyverno apply policies/ --resource test-resources/ --detailed-results
- name: Run tests
run: |
kyverno test policies/tests/
- name: Validate helm chart
run: |
helm lint kyverno-policies/
helm template kyverno-policies/ | kyverno apply - --resource test-resources/
deploy-staging:
needs: test
if: github.ref == 'refs/heads/main'
runs-on: ubuntu-latest
steps:
- name: Deploy to staging
run: |
helm upgrade --install kyverno-policies ./kyverno-policies \
-f values-staging.yaml \
--kubeconfig ${{ secrets.STAGING_KUBECONFIG }}