掃二維碼與項(xiàng)目經(jīng)理溝通
我們?cè)谖⑿派?4小時(shí)期待你的聲音
解答本文疑問(wèn)/技術(shù)咨詢(xún)/運(yùn)營(yíng)咨詢(xún)/技術(shù)建議/互聯(lián)網(wǎng)交流
如果你希望在 IP 地址或端口層面(OSI 第 3 層或第 4 層)控制網(wǎng)絡(luò)流量, 則你可以考慮為集群中特定應(yīng)用使用 Kubernetes 網(wǎng)絡(luò)策略(NetworkPolicy)。 NetworkPolicy 是一種以應(yīng)用為中心的結(jié)構(gòu),允許你設(shè)置如何允許 Pod 與網(wǎng)絡(luò)上的各類(lèi)網(wǎng)絡(luò)“實(shí)體” (我們這里使用實(shí)體以避免過(guò)度使用諸如“端點(diǎn)”和“服務(wù)”這類(lèi)常用術(shù)語(yǔ), 這些術(shù)語(yǔ)在 Kubernetes 中有特定含義)通信。

創(chuàng)新互聯(lián)堅(jiān)持“要么做到,要么別承諾”的工作理念,服務(wù)領(lǐng)域包括:成都網(wǎng)站制作、做網(wǎng)站、外貿(mào)營(yíng)銷(xiāo)網(wǎng)站建設(shè)、企業(yè)官網(wǎng)、英文網(wǎng)站、手機(jī)端網(wǎng)站、網(wǎng)站推廣等服務(wù),滿(mǎn)足客戶(hù)于互聯(lián)網(wǎng)時(shí)代的固安網(wǎng)站設(shè)計(jì)、移動(dòng)媒體設(shè)計(jì)的需求,幫助企業(yè)找到有效的互聯(lián)網(wǎng)解決方案。努力成為您成熟可靠的網(wǎng)絡(luò)建設(shè)合作伙伴!
Pod 可以通信的 Pod 是通過(guò)如下三個(gè)標(biāo)識(shí)符的組合來(lái)辯識(shí)的:
在定義基于 Pod 或名字空間的 NetworkPolicy 時(shí),你會(huì)使用 選擇算符 來(lái)設(shè)定哪些流量 可以進(jìn)入或離開(kāi)與該算符匹配的 Pod。
同時(shí),當(dāng)基于 IP 的 NetworkPolicy 被創(chuàng)建時(shí),我們基于 IP 組塊(CIDR 范圍) 來(lái)定義策略。
網(wǎng)絡(luò)策略通過(guò)網(wǎng)絡(luò)插件 來(lái)實(shí)現(xiàn)。要使用網(wǎng)絡(luò)策略,你必須使用支持 NetworkPolicy 的網(wǎng)絡(luò)解決方案。 創(chuàng)建一個(gè) NetworkPolicy 資源對(duì)象而沒(méi)有控制器來(lái)使它生效的話(huà),是沒(méi)有任何作用的。
Pod 有兩種隔離: 出口的隔離和入口的隔離。它們涉及到可以建立哪些連接。 這里的“隔離”不是絕對(duì)的,而是意味著“有一些限制”。 另外的,“非隔離方向”意味著在所述方向上沒(méi)有限制。這兩種隔離(或不隔離)是獨(dú)立聲明的, 并且都與從一個(gè) Pod 到另一個(gè) Pod 的連接有關(guān)。
默認(rèn)情況下,一個(gè) Pod 的出口是非隔離的,即所有外向連接都是被允許的。如果有任何的 NetworkPolicy 選擇該 Pod 并在其 ?policyTypes ?中包含 “Egress”,則該 Pod 是出口隔離的, 我們稱(chēng)這樣的策略適用于該 Pod 的出口。當(dāng)一個(gè) Pod 的出口被隔離時(shí), 唯一允許的來(lái)自 Pod 的連接是適用于出口的 Pod 的某個(gè) NetworkPolicy 的 ?egress ?列表所允許的連接。 這些 ?egress ?列表的效果是相加的。
默認(rèn)情況下,一個(gè) Pod 對(duì)入口是非隔離的,即所有入站連接都是被允許的。如果有任何的 NetworkPolicy 選擇該 Pod 并在其 ?policyTypes ?中包含 “Ingress”,則該 Pod 被隔離入口, 我們稱(chēng)這種策略適用于該 Pod 的入口。 當(dāng)一個(gè) Pod 的入口被隔離時(shí),唯一允許進(jìn)入該 Pod 的連接是來(lái)自該 Pod 節(jié)點(diǎn)的連接和適用于入口的 Pod 的某個(gè) NetworkPolicy 的 ?ingress ?列表所允許的連接。這些 ?ingress ?列表的效果是相加的。
網(wǎng)絡(luò)策略是相加的,所以不會(huì)產(chǎn)生沖突。如果策略適用于 Pod 某一特定方向的流量, Pod 在對(duì)應(yīng)方向所允許的連接是適用的網(wǎng)絡(luò)策略所允許的集合。 因此,評(píng)估的順序不影響策略的結(jié)果。
要允許從源 Pod 到目的 Pod 的連接,源 Pod 的出口策略和目的 Pod 的入口策略都需要允許連接。 如果任何一方不允許連接,建立連接將會(huì)失敗。
下面是一個(gè) NetworkPolicy 的示例:
apiVersion: networking.K8S.io/v1
kind: NetworkPolicy
metadata:
name: test-network-policy
namespace: default
spec:
podSelector:
matchLabels:
role: db
policyTypes:
- Ingress
- Egress
ingress:
- from:
- ipBlock:
cidr: 172.17.0.0/16
except:
- 172.17.1.0/24
- namespaceSelector:
matchLabels:
project: myproject
- podSelector:
matchLabels:
role: frontend
ports:
- protocol: TCP
port: 6379
egress:
- to:
- ipBlock:
cidr: 10.0.0.0/24
ports:
- protocol: TCP
port: 5978
除非選擇支持網(wǎng)絡(luò)策略的網(wǎng)絡(luò)解決方案,否則將上述示例發(fā)送到API服務(wù)器沒(méi)有任何效果。
必需字段:與所有其他的 Kubernetes 配置一樣,NetworkPolicy 需要 ?apiVersion?、 ?kind ?和 ?metadata ?字段。
spec:NetworkPolicy 規(guī)約 中包含了在一個(gè)名字空間中定義特定網(wǎng)絡(luò)策略所需的所有信息。
podSelector:每個(gè) NetworkPolicy 都包括一個(gè) ?podSelector?,它對(duì)該策略所 適用的一組 Pod 進(jìn)行選擇。示例中的策略選擇帶有 "role=db" 標(biāo)簽的 Pod。 空的 ?podSelector ?選擇名字空間下的所有 Pod。
policyTypes: 每個(gè) NetworkPolicy 都包含一個(gè) ?policyTypes ?列表,其中包含 ?Ingress ?或 ?Egress ?或兩者兼具。?policyTypes ?字段表示給定的策略是應(yīng)用于 進(jìn)入所選 Pod 的入站流量還是來(lái)自所選 Pod 的出站流量,或兩者兼有。 如果 NetworkPolicy 未指定 ?policyTypes ?則默認(rèn)情況下始終設(shè)置 ?Ingress?; 如果 NetworkPolicy 有任何出口規(guī)則的話(huà)則設(shè)置 ?Egress?。
ingress: 每個(gè) NetworkPolicy 可包含一個(gè) ?ingress ?規(guī)則的白名單列表。 每個(gè)規(guī)則都允許同時(shí)匹配 ?from ?和 ?ports ?部分的流量。示例策略中包含一條 簡(jiǎn)單的規(guī)則: 它匹配某個(gè)特定端口,來(lái)自三個(gè)來(lái)源中的一個(gè),第一個(gè)通過(guò) ?ipBlock ?指定,第二個(gè)通過(guò) ?namespaceSelector ?指定,第三個(gè)通過(guò) ?podSelector ?指定。
egress: 每個(gè) NetworkPolicy 可包含一個(gè) ?egress ?規(guī)則的白名單列表。 每個(gè)規(guī)則都允許匹配 ?to ?和 ?port ?部分的流量。該示例策略包含一條規(guī)則, 該規(guī)則將指定端口上的流量匹配到 ?10.0.0.0/24? 中的任何目的地。
所以,該網(wǎng)絡(luò)策略示例:
可以在 ?ingress ?的 ?from ?部分或 ?egress ?的 ?to ?部分中指定四種選擇器:
podSelector: 此選擇器將在與 NetworkPolicy 相同的名字空間中選擇特定的 Pod,應(yīng)將其允許作為入站流量來(lái)源或出站流量目的地。
namespaceSelector:此選擇器將選擇特定的名字空間,應(yīng)將所有 Pod 用作其 入站流量來(lái)源或出站流量目的地。
namespaceSelector 和 podSelector: 一個(gè)指定 ?namespaceSelector ?和 ?podSelector ?的 ?to?/?from ?條目選擇特定名字空間中的特定 Pod。 注意使用正確的 YAML 語(yǔ)法;下面的策略:
...
ingress:
- from:
- namespaceSelector:
matchLabels:
user: alice
podSelector:
matchLabels:
role: client
...在 ?from ?數(shù)組中僅包含一個(gè)元素,只允許來(lái)自標(biāo)有 ?role=client? 的 Pod 且 該 Pod 所在的名字空間中標(biāo)有 ?user=alice? 的連接。但是 這項(xiàng) 策略:
...
ingress:
- from:
- namespaceSelector:
matchLabels:
user: alice
- podSelector:
matchLabels:
role: client
...在 ?from ?數(shù)組中包含兩個(gè)元素,允許來(lái)自本地名字空間中標(biāo)有 ?role=client? 的 Pod 的連接,或 來(lái)自任何名字空間中標(biāo)有 ?user=alice? 的任何 Pod 的連接。
如有疑問(wèn),請(qǐng)使用 ?kubectl describe? 查看 Kubernetes 如何解釋該策略。
ipBlock: 此選擇器將選擇特定的 IP CIDR 范圍以用作入站流量來(lái)源或出站流量目的地。 這些應(yīng)該是集群外部 IP,因?yàn)?nbsp;Pod IP 存在時(shí)間短暫的且隨機(jī)產(chǎn)生。
集群的入站和出站機(jī)制通常需要重寫(xiě)數(shù)據(jù)包的源 IP 或目標(biāo) IP。 在發(fā)生這種情況時(shí),不確定在 NetworkPolicy 處理之前還是之后發(fā)生, 并且對(duì)于網(wǎng)絡(luò)插件、云提供商、?Service ?實(shí)現(xiàn)等的不同組合,其行為可能會(huì)有所不同。
對(duì)入站流量而言,這意味著在某些情況下,你可以根據(jù)實(shí)際的原始源 ?IP ?過(guò)濾傳入的數(shù)據(jù)包, 而在其他情況下,NetworkPolicy 所作用的 源IP 則可能是 ?LoadBalancer ?或 Pod 的節(jié)點(diǎn)等。
對(duì)于出站流量而言,這意味著從 Pod 到被重寫(xiě)為集群外部 IP 的 ?Service ?IP 的連接可能會(huì)或可能不會(huì)受到基于 ?ipBlock ?的策略的約束
默認(rèn)情況下,如果名字空間中不存在任何策略,則所有進(jìn)出該名字空間中 Pod 的流量都被允許。 以下示例使你可以更改該名字空間中的默認(rèn)行為。
你可以通過(guò)創(chuàng)建選擇所有容器但不允許任何進(jìn)入這些容器的入站流量的 NetworkPolicy 來(lái)為名字空間創(chuàng)建 “default” 隔離策略。
apiVersion: networking.k8s.io/v1
kind: NetworkPolicy
metadata:
name: default-deny-ingress
spec:
podSelector: {}
policyTypes:
- Ingress
這樣可以確保即使容器沒(méi)有選擇其他任何 NetworkPolicy,也仍然可以被隔離。 此策略不會(huì)更改默認(rèn)的出口隔離行為。
如果要允許所有流量進(jìn)入某個(gè)名字空間中的所有 Pod(即使添加了導(dǎo)致某些 Pod 被視為 “隔離”的策略),則可以創(chuàng)建一個(gè)策略來(lái)明確允許該名字空間中的所有流量。
apiVersion: networking.k8s.io/v1
kind: NetworkPolicy
metadata:
name: allow-all-ingress
spec:
podSelector: {}
ingress:
- {}
policyTypes:
- Ingress
你可以通過(guò)創(chuàng)建選擇所有容器但不允許來(lái)自這些容器的任何出站流量的 NetworkPolicy 來(lái)為名字空間創(chuàng)建 “default” 隔離策略。
apiVersion: networking.k8s.io/v1
kind: NetworkPolicy
metadata:
name: default-deny-egress
spec:
podSelector: {}
policyTypes:
- Egress
此策略可以確保即使沒(méi)有被其他任何 NetworkPolicy 選擇的 Pod 也不會(huì)被允許流出流量。 此策略不會(huì)更改默認(rèn)的入站流量隔離行為。
如果要允許來(lái)自名字空間中所有 Pod 的所有流量(即使添加了導(dǎo)致某些 Pod 被視為“隔離”的策略), 則可以創(chuàng)建一個(gè)策略,該策略明確允許該名字空間中的所有出站流量。
apiVersion: networking.k8s.io/v1
kind: NetworkPolicy
metadata:
name: allow-all-egress
spec:
podSelector: {}
egress:
- {}
policyTypes:
- Egress
你可以為名字空間創(chuàng)建“默認(rèn)”策略,以通過(guò)在該名字空間中創(chuàng)建以下 NetworkPolicy 來(lái)阻止所有入站和出站流量。
apiVersion: networking.k8s.io/v1
kind: NetworkPolicy
metadata:
name: default-deny-all
spec:
podSelector: {}
policyTypes:
- Ingress
- Egress
此策略可以確保即使沒(méi)有被其他任何 NetworkPolicy 選擇的 Pod 也不會(huì)被 允許入站或出站流量。
FEATURE STATE: Kubernetes v1.20 [stable]
作為一個(gè)穩(wěn)定特性,SCTP 支持默認(rèn)是被啟用的。 要在集群層面禁用 SCTP,你(或你的集群管理員)需要為 API 服務(wù)器指定 ?--feature-gates=SCTPSupport=false,...? 來(lái)禁用 ?SCTPSupport ?特性門(mén)控。 啟用該特性門(mén)控后,用戶(hù)可以將 NetworkPolicy 的 ?protocol ?字段設(shè)置為 ?SCTP?。
你必須使用支持 SCTP 協(xié)議網(wǎng)絡(luò)策略的 CNI 插件。
FEATURE STATE: Kubernetes v1.22 [beta]
在編寫(xiě) NetworkPolicy 時(shí),你可以針對(duì)一個(gè)端口范圍而不是某個(gè)固定端口。
這一目的可以通過(guò)使用 ?endPort ?字段來(lái)實(shí)現(xiàn),如下例所示:
apiVersion: networking.k8s.io/v1
kind: NetworkPolicy
metadata:
name: multi-port-egress
namespace: default
spec:
podSelector:
matchLabels:
role: db
policyTypes:
- Egress
egress:
- to:
- ipBlock:
cidr: 10.0.0.0/24
ports:
- protocol: TCP
port: 32000
endPort: 32768上面的規(guī)則允許名字空間 ?default ?中所有帶有標(biāo)簽 ?role=db ?的 Pod 使用 TCP 協(xié)議 與 ?10.0.0.0/24? 范圍內(nèi)的 IP 通信,只要目標(biāo)端口介于 32000 和 32768 之間就可以。
使用此字段時(shí)存在以下限制:
endPort ?字段,你(或者你的集群管理員)需要為 API 服務(wù)器設(shè)置 ?-feature-gates=NetworkPolicyEndPort=false,... ?以禁用 ?NetworkPolicyEndPort ?特性門(mén)控。endPort ?字段必須等于或者大于 ?port ?字段的值。你的集群所使用的 CNI 插件 必須支持在 NetworkPolicy 規(guī)約中使用 ?
endPort?字段。 如果你的網(wǎng)絡(luò)插件 不支持 ?endPort?字段,而你指定了一個(gè)包含 ?endPort?字段的 NetworkPolicy, 策略只對(duì)單個(gè) ?port?字段生效。
FEATURE STATE: Kubernetes 1.22 [stable]
只要 ?NamespaceDefaultLabelName ?特性門(mén)控 被啟用,Kubernetes 控制面會(huì)在所有名字空間上設(shè)置一個(gè)不可變更的標(biāo)簽 ?kubernetes.io/metadata.name?。該標(biāo)簽的值是名字空間的名稱(chēng)。
如果 NetworkPolicy 無(wú)法在某些對(duì)象字段中指向某名字空間,你可以使用標(biāo)準(zhǔn)的 標(biāo)簽方式來(lái)指向特定名字空間。
到 Kubernetes 1.24 為止,NetworkPolicy API 還不支持以下功能,不過(guò) 你可能可以使用操作系統(tǒng)組件(如 SELinux、OpenVSwitch、IPTables 等等) 或者第七層技術(shù)(Ingress 控制器、服務(wù)網(wǎng)格實(shí)現(xiàn))或準(zhǔn)入控制器來(lái)實(shí)現(xiàn)一些 替代方案。 如果你對(duì) Kubernetes 中的網(wǎng)絡(luò)安全性還不太了解,了解使用 NetworkPolicy API 還無(wú)法實(shí)現(xiàn)下面的用戶(hù)場(chǎng)景是很值得的。

我們?cè)谖⑿派?4小時(shí)期待你的聲音
解答本文疑問(wèn)/技術(shù)咨詢(xún)/運(yùn)營(yíng)咨詢(xún)/技術(shù)建議/互聯(lián)網(wǎng)交流