init
This commit is contained in:
@@ -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 }}
|
||||||
Reference in New Issue
Block a user