Kubernetes安全之KubeEye

語言: CN / TW / HK

一 前言

KubeEye 是一款 Kubernetes 安全及配置問題檢測工具,針對部署在 K8s 叢集中的業務應用進行配置檢測使用 OPA,針對叢集部署的 Node 使用Node-Problem-Detector進行檢測,同時除了系統內建有根據大多數業界常見場景的預定義規則,還支援使用者自定義規則來進行叢集檢測。

二 架構

KubeEye 通過呼叫 Kubernetes API,通過匹配資源中的關鍵字和容器語法的規則匹配來獲取叢集診斷資料,詳見架構圖。

其中針對 Node 節點檢測,需要在被檢測 Node 主機上安裝

三 特點

3.1 特性

  • KubeEye 根據行業最佳實踐審查你的工作負載 yaml 規範,幫助你使你的叢集穩定。
  • KubeEye 可以發現你的叢集控制平面的問題,包括 kube-apiserver/kube-controller-manager/etcd 等。
  • KubeEye 可以幫助你檢測各種節點問題,包括記憶體/CPU/磁碟壓力,意外的核心錯誤日誌等。

3.2 檢查項

| 是/否 | 檢查項 | 描述 | 級別 | | ----- | -------------------------- | ---------------------------------------------- | ---- | | ✅ | PrivilegeEscalationAllowed | 允許特權升級 | 緊急 | | ✅ | CanImpersonateUser | role/clusterrole 有偽裝成其他使用者許可權 | 警告 | | ✅ | CanDeleteResources | role/clusterrole 有刪除 kubernetes 資源許可權 | 警告 | | ✅ | CanModifyWorkloads | role/clusterrole 有修改 kubernetes 資源許可權 | 警告 | | ✅ | NoCPULimits | 資源沒有設定 CPU 使用限制 | 緊急 | | ✅ | NoCPURequests | 資源沒有設定預留 CPU | 緊急 | | ✅ | HighRiskCapabilities | 開啟了高危功能,例如 ALL/SYS_ADMIN/NET_ADMIN | 緊急 | | ✅ | HostIPCAllowed | 開啟了主機 IPC | 緊急 | | ✅ | HostNetworkAllowed | 開啟了主機網路 | 緊急 | | ✅ | HostPIDAllowed | 開啟了主機PID | 緊急 | | ✅ | HostPortAllowed | 開啟了主機埠 | 緊急 | | ✅ | ImagePullPolicyNotAlways | 映象拉取策略不是 always | 警告 | | ✅ | ImageTagIsLatest | 映象標籤是 latest | 警告 | | ✅ | ImageTagMiss | 映象沒有標籤 | 緊急 | | ✅ | InsecureCapabilities | 開啟了不安全的功能,例如 KILL/SYS_CHROOT/CHOWN | 警告 | | ✅ | NoLivenessProbe | 沒有設定存活狀態檢查 | 警告 | | ✅ | NoMemoryLimits | 資源沒有設定記憶體使用限制 | 緊急 | | ✅ | NoMemoryRequests | 資源沒有設定預留記憶體 | 緊急 | | ✅ | NoPriorityClassName | 沒有設定資源排程優先順序 | 通知 | | ✅ | PrivilegedAllowed | 以特權模式執行資源 | 緊急 | | ✅ | NoReadinessProbe | 沒有設定就緒狀態檢查 | 警告 | | ✅ | NotReadOnlyRootFilesystem | 沒有設定根檔案系統為只讀 | 警告 | | ✅ | NotRunAsNonRoot | 沒有設定禁止以 root 使用者啟動程序 | 警告 | | ✅ | CertificateExpiredPeriod | 將檢查 ApiServer 證書的到期日期少於30天 | 緊急 | | ✅ | EventAudit | 事件檢查 | 警告 | | ✅ | NodeStatus | 節點狀態檢查 | 警告 | | ✅ | DockerStatus | docker 狀態檢查 | 警告 | | ✅ | KubeletStatus | kubelet 狀態檢查 | 警告 |

四 部署

kubeEye 本身使用 Golang 編寫,可使用編譯好的二進位制可執行檔案進行相關元件安裝。

4.1 安裝

4.1.1 二進位制安裝

shell wget https://github.com/kubesphere/kubeeye/releases/download/v0.3.0/kubeeye-0.3.0-linux-amd64.tar.gz tar -zxvf kubeeye-0.3.0-linux-amd64.tar.gz mv kubeeye /usr/bin/

4.1.2 原始碼編譯安裝

shell git clone https://github.com/kubesphere/kubeeye.git cd kubeeye make installke

4.2 安裝NPD

針對叢集 Node 主機的檢測,kubeEye 採用Node-problem-Detector ,需要在 Node 主機節點進行安裝,kubeeye 封裝安裝命令,可以進行一鍵安裝。

⚠️注意:這將在你的叢集上安裝 npd,只有當你想要詳細的節點報告時才需要。

[[email protected] ~]# kubeeye install -e npd kube-system ConfigMap node-problem-detector-config created kube-system DaemonSet node-problem-detector created

其主要在 kube-system 名稱空間建立 node-problem-detector-config 的 ConfigMap 和 node-problem-detector DaemonSet。

4.3 叢集中執行kubeEye

kubeEye 除了可以一次性使用工具執行,同時 kubeEye 也是一個 Operator,可以執行在叢集內部,進行長久的持續對叢集進行檢測。

4.3.1 Kubernetes 中部署 Kubeeye

shell kubectl apply -f https://raw.githubusercontent.com/kubesphere/kubeeye/main/deploy/kubeeye.yaml kubectl apply -f https://raw.githubusercontent.com/kubesphere/kubeeye/main/deploy/kubeeye_insights.yaml

4.3.2 檢視 Kubeeye 巡檢結果

```shell $ kubectl get clusterinsight -o yaml

apiVersion: v1 items: - apiVersion: kubeeye.kubesphere.io/v1alpha1 kind: ClusterInsight metadata: name: clusterinsight-sample namespace: default spec: auditPeriod: 24h status: auditResults: auditResults: - resourcesType: Node resultInfos: - namespace: "" resourceInfos: - items: - level: waring message: KubeletHasNoSufficientMemory reason: kubelet has no sufficient memory available - level: waring message: KubeletHasNoSufficientPID reason: kubelet has no sufficient PID available - level: waring message: KubeletHasDiskPressure reason: kubelet has disk pressure name: kubeeyeNode ```

五 測試

5.1 命令選項

```shell [[email protected] ~]# kubeeye -h KubeEye finds various problems on Kubernetes cluster.

Usage: ke [command]

Available Commands: audit audit resources from the cluster completion generate the autocompletion script for the specified shell help Help about any command install A brief description of your command uninstall A brief description of your command

Flags: -f, --config string Specify the path of kubeconfig. -h, --help help for ke --kubeconfig string Paths to a kubeconfig. Only required if out-of-cluster. --master --kubeconfig (Deprecated: switch to --kubeconfig) The address of the Kubernetes API server. Overrides any value in kubeconfig. Only required if out-of-cluster. ```

可以看到 kubeeye 目前主要支援兩個命令,一個為 install package 例如 NPD,另外一個執行 audit,對叢集應用進行配置掃描。

5.2 audit

shell [[email protected] ~]# kubeeye audit KIND NAMESPACE NAME MESSAGE Deployment dddd jenkins-1644220286 [NoCPULimits ImagePullPolicyNotAlways NoMemoryLimits NoPriorityClassName NotReadOnlyRootFilesystem NotRunAsNonRoot] Deployment jenkins jenkins-1644220286 [NoCPULimits ImagePullPolicyNotAlways NoMemoryLimits NoPriorityClassName NotReadOnlyRootFilesystem NotRunAsNonRoot] Deployment smartkm-api-k8s velero [ImageTagIsLatest NoLivenessProbe NoPriorityClassName NotReadOnlyRootFilesystem NoReadinessProbe NotRunAsNonRoot] DaemonSet smartkm-api-k8s restic [ImageTagIsLatest NoLivenessProbe NoPriorityClassName NotReadOnlyRootFilesystem NoReadinessProbe NotRunAsNonRoot] Node minikube [KernelHasNoDeadlock FilesystemIsNotReadOnly KubeletHasSufficientMemory KubeletHasNoDiskPressure KubeletHasSufficientPID] Event kube-system node-problem-detector-dmsws.16d844532f662318 [Failed to pull image "k8s.gcr.io/node-problem-detector/node-problem-detector:v0.8.7": rpc error: code = Unknown desc = Error response from daemon: Get https://k8s.gcr.io/v2/: net/http: request canceled while waiting for connection (Client.Timeout exceeded while awaiting headers)] Event kube-system node-problem-detector-dmsws.16d844532f66703e [Error: ErrImagePull] Event kube-system node-problem-detector-dmsws.16d84453351b8b19 [Error: ImagePullBackOff]

5.3 新增自定義檢查規則

我們利用命令檢視預定義 OPA 檢查規則。

shell kubectl get cm -n kube-system node-problem-detector-config -oyaml

同時也可以根據自己業務建立自定義檢查規則。

  • 建立 OPA 規則存放目錄

mkdir opa

  • 新增自定義 OPA 規則檔案

注意:為檢查工作負載設定的 OPA 規則, package 名稱必須是 kubeeye_workloads_rego 為檢查 RBAC 設定的 OPA 規則, package 名稱必須是 kubeeye_RBAC_rego 為檢查節點設定的 OPA 規則, package 名稱必須是 kubeeye_nodes_rego

  • 以下為檢查映象倉庫地址規則,儲存以下規則到規則檔案 imageRegistryRule.rego

``` package kubeeye_workloads_rego

deny[msg] { resource := input type := resource.Object.kind resourcename := resource.Object.metadata.name resourcenamespace := resource.Object.metadata.namespace workloadsType := {"Deployment","ReplicaSet","DaemonSet","StatefulSet","Job"} workloadsType[type]

not workloadsImageRegistryRule(resource)

msg := {
    "Name": sprintf("%v", [resourcename]),
    "Namespace": sprintf("%v", [resourcenamespace]),
    "Type": sprintf("%v", [type]),
    "Message": "ImageRegistryNotmyregistry"
}

}

workloadsImageRegistryRule(resource) { regex.match("^myregistry.public.kubesphere/basic/.+", resource.Object.spec.template.spec.containers[_].image) } ```

  • 使用額外的規則執行 kubeeye

提示:kubeeye 將讀取指定目錄下所有 .rego 結尾的檔案

kubeeye audit -p ./opa

六 問題排查

  • NPD 安裝異常,預設使用 k8s.gcr.io,如果安裝伺服器無法連通公網可使用我的映象倉庫:1832990/node-problem-detector:v0.8.7。
  • kubEye 安裝使用預設使用主機$HOME/.kube/config 檔案,如果不存在 K8s config 檔案,則無法正常執行。

參考連結

  • https://github.com/kubesphere/kubeeye/

其他