微服务:用控制器来完成集群的工作负载,那么应用如何暴漏出去?需要通过微服务暴漏出去后才能被访问
-
Service是一组提供相同服务的Pod对外开放的接口。
-
借助Service,应用可以实现服务发现和负载均衡。
-
service默认只支持4层负载均衡能力,没有7层功能。(可以通过Ingress实现)
data:image/s3,"s3://crabby-images/978f3/978f3bba6a72fb3ed8f53f15a65cca2944e58757" alt=""
微服务类型
微服务类型 | 作用描述 |
| ------------ | ------------------------------------------------------------ |
| ClusterIP | 默认值,k8s系统给service自动分配的虚拟IP,只能在集群内部访问 |
| NodePort | 将Service通过指定的Node上的端口暴露给外部,访问任意一个NodeIP:nodePort都将路由到ClusterIP |
| LoadBalancer | 在NodePort的基础上,借助cloud provider创建一个外部的负载均衡器,并将请求转发到 NodeIP:NodePort,此模式只能在云服务器上使用 |
| ExternalName | 将服务通过 DNS CNAME 记录方式转发到指定的域名(通过 spec.externlName 设定 |
示例:
#生成控制器文件并建立控制器
kubectl create deployment timinglee --image myapp:v1 --replicas 2 --dry-run=client -o yaml > timinglee.yaml
生成微服务yaml追加到已有yaml中
data:image/s3,"s3://crabby-images/f5018/f5018ffade571151a9b39d22c35706d42bf76c3a" alt=""
kubectl expose deployment timinglee --port 80 --target-port 80 --dry-run=client -o yaml >> timinglee.yaml
vi timinglee.yaml
data:image/s3,"s3://crabby-images/21639/21639611cc1090346c6e4e4425fc5a2f4bda29ff" alt=""
data:image/s3,"s3://crabby-images/d12c9/d12c91e327bf7222615bd45b2a5e8d70e6e3f2ff" alt=""
data:image/s3,"s3://crabby-images/dc965/dc96585247dad1bfb90437a33b5818af80affb92" alt=""
查看策略
iptables -t nat -nL
data:image/s3,"s3://crabby-images/31412/31412c97cf77ed6e0d32f1d0333b0ea9b03e4e41" alt=""
IPVS模式
配置方式
在所有节点安装ipvsadm
yum install ipvsadm -y
data:image/s3,"s3://crabby-images/c31f3/c31f346c0887ae2f06f0f791cd4b2e0097adca5a" alt=""
修改master 节点的代理配置
kubectl -n kube-system edit cm kube-proxy
data:image/s3,"s3://crabby-images/9178d/9178d94a5eb2cb833c52b46152ab9c5085660b58" alt=""
修改后重启pod
data:image/s3,"s3://crabby-images/1e3d2/1e3d21bf68502dcc279b048adb9b66ef4c8d62a9" alt=""
data:image/s3,"s3://crabby-images/80bea/80bea7a59aaa53d9fa5e85db8ce00cfb4face736" alt=""
data:image/s3,"s3://crabby-images/d877f/d877f0cea3ad413ebcbaa29c058c4f9a03ada0e6" alt=""
切换ipvs模式后,kube-proxy会在宿主机上添加一个虚拟网卡:kube-ipvs0,并分配所有service IP
data:image/s3,"s3://crabby-images/b448b/b448b72c4c62182037db0d6e5d112d77f1c161e7" alt=""
微服务类型详解
clusterip
特点:
clusterip模式只能在集群内访问,并对集群内的pod提供健康检测和自动发现功能
示例
vi myapp.yml
data:image/s3,"s3://crabby-images/49d42/49d4250be59688fb25e83584f5dfcfcdd7e1472f" alt=""
service创建后集群DNS提供解析
data:image/s3,"s3://crabby-images/dd605/dd6056027ad6b62beb19f2edc47f2cd58f2e95d7" alt=""
ClusterIP中的特殊模式headless
headless(无头服务)
对于无头 `Services` 并不会分配 Cluster IP,kube-proxy不会处理它们, 而且平台也不会为它们进行负载均衡和路由,集群访问通过dns解析直接指向到业务pod上的IP,所有的调度有dns单独完成
vim timinglee.yaml
data:image/s3,"s3://crabby-images/73f57/73f5783427b1b7e577bdb77b2d5f89fbba1611b9" alt=""
测试
data:image/s3,"s3://crabby-images/42985/42985dde012efc4d739815b6c935957ab111cf47" alt=""
dig timinglee.default.svc.cluster.local @10.96.0.10
data:image/s3,"s3://crabby-images/a4ada/a4ada9fb8353befca50838af16227c37d3199a13" alt=""
nodeport
通过ipvs暴漏端口从而使外部主机通过master节点的对外ip:<port>来访问pod业务
其访问过程为:
data:image/s3,"s3://crabby-images/fab48/fab48e16164144af7cc6a32dcaebec3ae43224e1" alt=""
示例
vim timinglee.yaml
data:image/s3,"s3://crabby-images/3f2a4/3f2a445465d7bff20167bbac6e726c33a8ae24be" alt=""
在集群节点上绑定端口,一个端口对应一个服务
data:image/s3,"s3://crabby-images/b8f5a/b8f5a6aa86a69cc539d1fdf68307b2411695e441" alt=""
注意!!
nodeport默认端口
nodeport默认端口是30000-32767,超出会报错
如果需要使用这个范围以外的端口就需要特殊设定
vim /etc/kubernetes/manifests/kube-apiserver.yaml
- --service-node-port-range=30000-40000
这里需要注意的是
添加"--service-node-port-range=" 参数,端口范围可以自定义
修改后api-server会自动重启,等apiserver正常启动后才能操作集群
集群重启自动完成在修改完参数后全程不需要人为干预
loadbalancer
云平台会为我们分配vip并实现访问,如果是裸金属主机那么需要metallb来实现ip的分配
data:image/s3,"s3://crabby-images/4354a/4354afb10a9e74454957cb4c262d4059bb886df2" alt=""
示例
data:image/s3,"s3://crabby-images/58289/582892ed5a9421180fb49313613f19f2daa7e9a8" alt=""
默认无法分配外部访问
data:image/s3,"s3://crabby-images/ab25e/ab25ecad4a47db896c753e11b6252f7ff2896c2e" alt=""
LoadBalancer模式适用云平台,裸金属环境需要安装metallb提供支持
metalLB
官网:https://metallb.universe.tf/installation/
data:image/s3,"s3://crabby-images/9a561/9a5617caebe9a0c4c361dd50d2417150a8686c89" alt=""
功能 为LoadBalancer分配vip
部署方式
kubectl edit cm -n kube-system kube-proxy
设置ipvs模式
data:image/s3,"s3://crabby-images/730cc/730cc8e83586575a930ca445980ae7d4a383b4d8" alt=""
data:image/s3,"s3://crabby-images/d8aac/d8aac3968fd34e083da239ba9aead427f3d7ef2c" alt=""
data:image/s3,"s3://crabby-images/f83d9/f83d951995d738db3f17216ab7ec21544192e8f3" alt=""
上传metallb 镜像到harbor仓库
data:image/s3,"s3://crabby-images/c1b46/c1b46fb81bef3288f676c7e18ad652b57bb4ec68" alt=""
data:image/s3,"s3://crabby-images/c441f/c441f35e47aff5f9b832266e288830a9bfc06bda" alt=""
data:image/s3,"s3://crabby-images/58ba8/58ba8ab19fdf66fbbb55397c869f11bd8752fba6" alt=""
部署服务
data:image/s3,"s3://crabby-images/41611/416119573cbc6478f499ed13273f62b4dee4ac44" alt=""
data:image/s3,"s3://crabby-images/8b9bf/8b9bf83be148441c8827c7284dff6bb381d5e71d" alt=""
配置分配地址段
vim configmap.yml
data:image/s3,"s3://crabby-images/319db/319dbe52ba8c09edafda44e75b006cf0ea7ffeb6" alt=""
两个不同的kind中间必须加分割
data:image/s3,"s3://crabby-images/b1fcb/b1fcbc396d0cc5e4cd56e8402232b6639729c79f" alt=""
查看
data:image/s3,"s3://crabby-images/0b7e6/0b7e6b5a96417fd8e8a26bd1eb8574e42e6841b7" alt=""
通过分配地址从集群外访问服务
data:image/s3,"s3://crabby-images/e70fd/e70fd738b92a6c6dc9be78b8bd698eb8a7df6b6c" alt=""
externalname
开启services后,不会被分配IP,而是用dns解析CNAME固定域名来解决ip变化问题
一般应用于外部业务和pod沟通或外部业务迁移到pod内时
在应用向集群迁移过程中,externalname在过度阶段就可以起作用了。
集群外的资源迁移到集群时,在迁移的过程中ip可能会变化,但是域名+dns解析能完美解决此问题
示例
data:image/s3,"s3://crabby-images/7f538/7f538bbc60a50b1485b4eefa8379ff51e8bc2141" alt=""
data:image/s3,"s3://crabby-images/b6cd6/b6cd613fdef0041a64afa55ee734375ea4fb03f8" alt=""
ingress-nginx
官网:https://kubernetes.github.io/ingress-nginx/deploy/#bare-metal-clusters
功能
data:image/s3,"s3://crabby-images/b5be5/b5be5f674395e384f8f2421c0622c24699307a1f" alt=""
一种全局的、为了代理不同后端 Service 而设置的负载均衡服务,支持7层
Ingress由两部分组成:Ingress controller和Ingress服务
Ingress Controller 会根据你定义的 Ingress 对象,提供对应的代理能力。
业界常用的各种反向代理项目,比如 Nginx、HAProxy、Envoy、Traefik 等,都已经为Kubernetes 专门维护了对应的 Ingress Controller。
部署
下载文件
data:image/s3,"s3://crabby-images/c17c4/c17c45cce0697bf1d4a1fc87aac7ec0b7d364636" alt=""
上传镜像
data:image/s3,"s3://crabby-images/66d83/66d835d821becb6a4e48d9439567d2b152402c4d" alt=""
data:image/s3,"s3://crabby-images/feba1/feba17e8fd55415918d0e3a65b2c109fe02faafb" alt=""
安装ingress
vim deploy.yaml
data:image/s3,"s3://crabby-images/68968/689684f7cf11b57f022ce9382647f5573ca35b69" alt=""
data:image/s3,"s3://crabby-images/07fa5/07fa5d55f78d124c64d4ac2fbf70a9ae716acfe4" alt=""
data:image/s3,"s3://crabby-images/59de6/59de6493c2898e137946a114c4792f702d987e24" alt=""
kubectl apply -f deploy.yaml
data:image/s3,"s3://crabby-images/977d0/977d05b1b1170c7682101d3bd5fa953d09fbb982" alt=""
kubectl -n ingress-nginx get svc
data:image/s3,"s3://crabby-images/43738/43738bb938f900136654418a8ef4585fe9e157b4" alt=""
修改微服务为loadbalancer
kubectl -n ingress-nginx edit svc ingress-nginx-controller
data:image/s3,"s3://crabby-images/d8ab3/d8ab3e1567213a1e67aec2f3c42e5525434b047f" alt=""
data:image/s3,"s3://crabby-images/490b5/490b548a5effbf2fc3f548b6116f3a3bf351846a" alt=""
注意
在ingress-nginx-controller中看到的对外IP就是ingress最终对外开放的ip
测试
kubectl create ingress webcluster --rule '*/=timinglee-svc:80' --dry-run=client -o yaml > timinglee-ingress.yml
data:image/s3,"s3://crabby-images/fbe62/fbe6277e238272d64d554bb0c8647bebebf34569" alt=""
建立ingress控制器
data:image/s3,"s3://crabby-images/f70c4/f70c4283fb898315cd1302451d3a104e21567802" alt=""
data:image/s3,"s3://crabby-images/5963a/5963a6b09918d76ecc15a4143f7f993c6852ee81" alt=""
for n in {1..5}; do curl 172.25.254.50/hostname.html; done
ingress必须和输出的service资源处于同一namespace
高级用法
kubectl create deployment myapp-v1 --image myapp:v1 --dry-run=client -o yaml > myapp-v1.yaml
kubectl create deployment myapp-v2 --image myapp:v2 --dry-run=client -o yaml > myapp-v2.yaml
在文件中加入
data:image/s3,"s3://crabby-images/013bb/013bbfcfbebc73eab77a6565b0cc93e19cd7af02" alt=""
data:image/s3,"s3://crabby-images/9f95b/9f95b1b36b9a8dc0b3398df74b50ce5e9f7cc7e8" alt=""
data:image/s3,"s3://crabby-images/d145c/d145c0560a969b3a199d2dc15421c7f879426173" alt=""
建立ingress的yaml文件、
data:image/s3,"s3://crabby-images/7670a/7670a2804a72cea14f2ec8f7e7a731ebe35cd341" alt=""
data:image/s3,"s3://crabby-images/aac90/aac9040972ac1735684057d8feec87f9ded7e2d0" alt=""
data:image/s3,"s3://crabby-images/76f16/76f160c745af09831e5370cc918baa4e731fdffd" alt=""
#nginx.ingress.kubernetes.io/rewrite-target: / 的功能实现
data:image/s3,"s3://crabby-images/f8446/f8446797249f649dc9e3a480f8f9cf1e73e83411" alt=""
基于域名的访问
在测试主机中设定解析
vi /etc/hosts
172.25.254.50 www.timinglee.org myappv1.timinglee.org myappv2.timinglee.org
建立基于域名的yml文件
vim ingress2.yml
data:image/s3,"s3://crabby-images/09082/09082c351731279cd5486a52db5f527123ca7135" alt=""
利用文件建立ingress
在测试主机中测试
data:image/s3,"s3://crabby-images/b6a74/b6a7429e2fbd8a5eed6976f65bd05b18e8a6709f" alt=""
建立tls加密
注意
secret通常在kubernetes中存放敏感数据,他并不是一种加密方式
openssl req -newkey rsa:2048 -nodes -keyout tls.key -x509 -days 365 -subj "/CN=nginxsvc/O=nginxsvc" -out tls.crt
data:image/s3,"s3://crabby-images/3d6ee/3d6ee88f3dd26db6c893d129b09a6ae05df3fd5c" alt=""
建立ingress3基于tls认证的yml文件
data:image/s3,"s3://crabby-images/60c61/60c61fadd17e4dcc643d1312e2ca2282b2a548e0" alt=""
测试
data:image/s3,"s3://crabby-images/44c18/44c18730bbb84440bca95300303da86cd3550daa" alt=""
建立认证
下载安装
http-tools
data:image/s3,"s3://crabby-images/bc110/bc110c2857f42ce5ec7f9c4d90708f0a22db7850" alt=""
建立认证类型资源
data:image/s3,"s3://crabby-images/b24bd/b24bdfdcedaea1b38319297021bae2919aa98e23" alt=""
data:image/s3,"s3://crabby-images/4b5ea/4b5eacf0c626772f99100ec4b06091b25532b58d" alt=""
测试
data:image/s3,"s3://crabby-images/32dea/32dea0d9065e6ed5f0dfed0c9d22d471f63ac584" alt=""
重定向
data:image/s3,"s3://crabby-images/7bad4/7bad46fbe9c77d5daff50f6a3cd4691628e48d14" alt=""
describe
data:image/s3,"s3://crabby-images/09b30/09b30cb3b63fbf2497e4b78b24a309c9691a5ef5" alt=""
测试
data:image/s3,"s3://crabby-images/447ea/447ea178146b18fb0e08c5bc022ca6fcdfd23b3e" alt=""
金丝雀发布
金丝雀发布(Canary Release)也称为灰度发布,是一种软件发布策略。
主要目的是在将新版本的软件全面推广到生产环境之前,先在一小部分用户或服务器上进行测试和验证,以降低因新版本引入重大问题而对整个系统造成的影响。
是一种Pod的发布方式。金丝雀发布采取先添加、再删除的方式,保证Pod的总量不低于期望值。并且在更新部分Pod后,暂停更新,当确认新Pod版本运行正常后再进行其他版本的Pod的更新。
data:image/s3,"s3://crabby-images/2d848/2d848b9792f3089b5364265df74b66db2ee9b1de" alt=""
canary发布方式
data:image/s3,"s3://crabby-images/1867f/1867fab447d969bd5e12bad2f2fae040b1a58829" alt=""
基于header(http包头)灰度
data:image/s3,"s3://crabby-images/4d05b/4d05be8297417f1709752d27e57cb2ec141e1c93" alt=""
通过Annotaion扩展
-
创建灰度ingress,配置灰度头部key以及value
-
灰度流量验证完毕后,切换正式ingress到新版本
-
之前我们在做升级时可以通过控制器做滚动更新,默认25%利用header可以使升级更为平滑,通过key 和vule 测试新的业务体系是否有问题。
示例
apiVersion: networking.k8s.io/v1
kind: Ingress
metadata:
annotations:
name: myapp-v1-ingress
spec:
ingressClassName: nginx
rules:
- host: myapp.timinglee.org
http:
paths:
- backend:
service:
name: myapp-v1
port:
number: 80
path: /
pathType: Prefix
建立基于header的ingress
apiVersion: networking.k8s.io/v1
kind: Ingress
metadata:
annotations:
nginx.ingress.kubernetes.io/canary: "true"
nginx.ingress.kubernetes.io/canary-by-header: version
nginx.ingress.kubernetes.io/canary-by-header-value: 2
name: myapp-v2-ingress
spec:
ingressClassName: nginx
rules:
- host: myapp.timinglee.org
http:
paths:
- backend:
service:
name: myapp-v2
port:
number: 80
path: /
pathType: Prefix
基于权重的灰度发布
data:image/s3,"s3://crabby-images/524f4/524f44ef5076048224edd3b62981def3b6b839b5" alt=""
apiVersion: networking.k8s.io/v1
kind: Ingress
metadata:
annotations:
nginx.ingress.kubernetes.io/canary: "true"
nginx.ingress.kubernetes.io/canary-weight: "10" #更改权重值
nginx.ingress.kubernetes.io/canary-weight-total: "100"
name: myapp-v2-ingress
spec:
ingressClassName: nginx
rules:
- host: myapp.timinglee.org
http:
paths:
- backend:
service:
name: myapp-v2
port:
number: 80
path: /
pathType: Prefix