プラクティス集@Istio¶
はじめに¶
本サイトにつきまして、以下をご認識のほど宜しくお願いいたします。
01. セットアップ¶
少ないコントロールプレーン¶
クラウドプロバイダーでIstioを稼働させる場合、各AZや各リージョンにコントロールプレーンを1
個だけセットアップし、できるだけ多くのアプリコンテナのサービスメッシュとなるようにする。
冗長化¶
コントロールプレーンの可用性を高めるために、コントロールプレーンを異なるAZに冗長化させる。
サービスメッシュに登録しないPodの選定 (サイドカーパターンの場合)¶
▼ 監視系のPod¶
サイドカーパターンに登録するPodが増えると、その分istio-proxy
コンテナが増える。
そのため、Pod当たりのハードウェアリソースの消費量が増えてしまう。
テレメトリーを収集する必要のないPod (例:監視を責務に持つPod) は、サイドカーパターンに登録しないようにする。
▼ Job配下のPod¶
Job配下のPodにistio-proxy
コンテナを挿入した場合、Pod内のコンテナが終了してもistio-proxy
コンテナが終了せず、Pod自体が削除されない問題がある。
Job配下のPodは、サイドカーパターンに登録しないようにする。
どうしてもサービスメッシュに登録したい場合は、Pod内のコンテナで、istio-proxy
コンテナの『localhost:15020/quitquitquit
』をコールするようなシェルスクリプトを実行する。
apiVersion: batch/v1
kind: Job
metadata:
name: foo-job
spec:
template:
metadata:
name: foo-job
spec:
containers:
- name: foo
command:
- /bin/bash
- -c
args:
- >
until curl -fsI http://localhost:15021/healthz/ready; do
echo "Waiting for Sidecar to be healthy";
sleep 3;
done;
echo "Sidecar available. Running job command..." &&
<CronJobのコマンド> &&
x=$(echo $?) &&
curl -fsI -X POST http://localhost:15020/quitquitquit &&
exit $x
02. トラフィック管理¶
Istio IngressGatewayに関して¶
▼ Istiodコントロールプレーンとは異なるNamespaceにおく¶
セキュリティ上の理由から、Istio IngressGatewayとIstiodコントロールプレーンは異なるNamespaceにおく方が良い。
▼ NodePort Serviceを選ぶ¶
Istio IngressGatewayでは、内部的に作成されるServiceのタイプ (NodePort Service、ClusterIP Service、LoadBalancer Service) を選べる。
NodePort Serviceを選ぶ場合、Nodeのダウンストリームに開発者がロードバランサーを作成し、NodePort Serviceにインバウンド通信をルーティングできるようにする。
パブリックネットワーク
⬇⬆︎︎
AWS Route53
⬇⬆︎︎
# L7ロードバランサー (単一のL7ロードバランサーを作成し、異なるポートを開放する複数のL4ロードバランサーの振り分ける)
AWS ALB
⬇⬆︎︎
# L4ロードバランサー
NodePort Service (Istio IngressGateway)
⬇⬆︎︎
Gateway
⬇⬆︎︎
VirtualService
⬇⬆︎︎
# L4ロードバランサー
ClusterIP Service
⬇⬆︎︎
Pod
一方で、LoadBalancer Serviceを選ぶ場合、クラウドプロバイダーのロードバランサーが自動的に作成される。
そのため、このロードバランサーからLoadBalancer Serviceにルーティングできるようにする。
LoadBalancer Serviceでは、クラウドプロバイダーのリソースとKubernetesリソースの責務の境界が曖昧になってしまうため、NodePort Serviceを選ぶようにする。
補足として、デフォルトではIstio IngressGatewayの内部ではLoadBalancer Serviceを作成されてしまう。
NodePort Serviceを選ぶためには、Istio IngressGatewayではなく、IstioOperatorやistioチャート上でServiceのタイプを設定し、Istio IngressGatewayを作成する必要がある。
▼ ClusterIP Serviceを選ぶ (AWS Load Balancer Controllerを使用する場合のみ)¶
AWS Load Balancer Controllerを使用する場合、Istio IngressGatewayでClusterIP Serviceを使用できる。
Ingressにて、alb.ingress.kubernetes.io/target-type
キー値をip
とすると、AWS Load Balancer ControllerはPodにリクエストを直接的にL7ロードバランシングする。
パブリックネットワーク
⬇⬆︎︎
AWS Route53
⬇⬆︎︎
# L7ロードバランサー (単一のL7ロードバランサーを作成し、異なるポートを開放する複数のL4ロードバランサーの振り分ける)
AWS Load Balancer ControllerによるAWS ALB
⬇⬆︎︎
# L4ロードバランサー
ClusterIP Service (Istio IngressGateway)
⬇⬆︎︎
Gateway
⬇⬆︎︎
VirtualService
⬇⬆︎︎
# L4ロードバランサー
ClusterIP Service
⬇⬆︎︎
Pod
▼ アプリコンテナごとに作成する¶
単一障害点になることを防ぐために、1
個のIstio IngressGatewayで全てのアプリコンテナにルーティングするのではなく、アプリコンテナことに用意する。
サブセット名を1
個にする¶
Istioリソースで設定するサブセット名は1
個だけにする。
これにより、Istio IngressGatewayで受信した通信を、特定のバージョンのPodにルーティングできる。
apiVersion: networking.istio.io/v1alpha3
kind: VirtualService
metadata:
name: foo-virtual-service
spec:
hosts:
# 特定のマイクロサービスへのリクエストのみを扱うため、ホスト名もそれのみを許可する
# ただし、gatewaysオプションがあるVirtualServiceではワイルドカードする
- foo
http:
- route:
- destination:
host: foo
subset: v1
Istioリソースのリクエスト可能な範囲を限定する¶
Istioリソースの.spec.exportTo
キーでは『.
(ドット) 』を設定する。
これにより、同じNamespaceからしかリクエストを受信できないようにする。
apiVersion: networking.istio.io/v1alpha3
kind: VirtualService
metadata:
name: foo-virtual-service
spec:
exportTo:
- "."
# 特定のマイクロサービスへのリクエストのみを扱うため、ホスト名もそれのみを許可する
# ただし、gatewaysオプションがあるVirtualServiceではワイルドカードする
hosts:
- foo
http:
- route:
- destination:
host: foo
DestinationRuleを最初に更新する¶
新しいサブセットを追加する場合、DestinationRuleを最初に更新する。
これにより、ダウンタイムなしでサブセットを追加できる。
DestinationRuleを更新する前に新しいサブセットを持つVirtualServiceを更新してしまうと、VirtualServiceは新しいサブセットを持つDestinationRuleを見つけられず、503
ステータスを返信してしまう。
DestinationRuleを最初に更新し、正常に完了することを待機した後に、VirtualServiceを更新する。
02-02. 通信ルーティングのパターン¶
LoadBalancer Serviceの場合¶
LoadBalancer Serviceを使用する場合、以下のようなネットワーク経路がある。
*例*
パブリックネットワーク
⬇⬆︎︎
AWS Route53
⬇⬆︎︎
# L4ロードバランサー
LoadBalancer Service (Istio IngressGateway) によるAWS NLB
⬇⬆︎︎
Gateway
⬇⬆︎︎
VirtualService
⬇⬆︎︎
# L4ロードバランサー
ClusterIP Service
⬇⬆︎︎
Pod
NodePort Serviceの場合¶
▼ L7
ロードバランサーがない場合¶
NodeのNICの宛先情報は、Node外から宛先IPアドレスとして指定できるため、インバウンド通信にIngressを必要としない。
*例*
パブリックネットワーク
⬇⬆︎︎
# L4ロードバランサー
NodePort Service (Istio IngressGateway)
⬇⬆︎︎
Gateway
⬇⬆︎︎
VirtualService
⬇⬆︎︎
# L4ロードバランサー
ClusterIP Service
⬇⬆︎︎
Pod
▼ L7
ロードバランサーがある場合¶
パブリックプロバイダーのロードバランサー (例:AWS ALB) を別に配置する。
単一のL7ロードバランサーを作成し、複数のL4ロードバランサーに振り分けるとよい。
*例*
パブリックネットワーク
⬇⬆︎︎
AWS Route53
⬇⬆︎︎
# L7ロードバランサー (単一のL7ロードバランサーを作成し、異なるポートを開放する複数のL4ロードバランサーの振り分ける)
AWS ALB
⬇⬆︎︎
# L4ロードバランサー
NodePort Service (Istio IngressGateway)
⬇⬆︎︎
Gateway
⬇⬆︎︎
VirtualService
⬇⬆︎︎
# L4ロードバランサー
ClusterIP Service
⬇⬆︎︎
Pod
ClusterIP Serviceの場合¶
AWS Load Balancer Controllerを使用する場合、Istio IngressGatewayでClusterIP Serviceを使用できる。
Ingressにて、alb.ingress.kubernetes.io/target-type
キー値をip
とすると、AWS Load Balancer ControllerはPodにリクエストを直接的にL7ロードバランシングする。
パブリックネットワーク
⬇⬆︎︎
AWS Route53
⬇⬆︎︎
# L7ロードバランサー (単一のL7ロードバランサーを作成し、異なるポートを開放する複数のL4ロードバランサーの振り分ける)
AWS Load Balancer ControllerによるAWS ALB
⬇⬆︎︎
# L4ロードバランサー
ClusterIP Service (Istio IngressGateway)
⬇⬆︎︎
Gateway
⬇⬆︎︎
VirtualService
⬇⬆︎︎
# L4ロードバランサー
ClusterIP Service
⬇⬆︎︎
Pod
03. アップグレード¶
設計規約¶
▼ サポート期間¶
Istioでは、マイナーバージョンごとのアップグレードを推奨しており、またマイナーバージョンのサポートが半年ごとに終了する。
実質的に半年ごとにアップグレード工数が発生する。
▼ マイナーバージョン単位でアップグレード¶
Istioの開発プロジェクトでは、マイナーバージョンを1
個ずつ新しくするアップグレードしか検証していない。
そのため、マイナーバージョンを2
個以上跨いだアップグレードを推奨していない。
▼ Istiodコントロールプレーンでダウンタイムを発生させない¶
Istiodコントロールプレーンでダウンタイムが発生すると、istio-proxy
コンテナ内のpilot-agentが最新の宛先情報を取得できなくなる。
そのため、古いバージョンのアプリコンテナの宛先情報を使用してしまう。
Istiodコントロールプレーンをカナリアアップグレードを採用する。
▼ Istio IngressGatewayでダウンタイムを発生させない¶
Istio IngressGatewayでダウンタイムが発生すると、アプリへのインバウンド通信が遮断されてしまう。
インプレース方式¶
▼ インプレース方式とは¶
既存のIstiodコントロールプレーンとIstio IngressGatewayの両方をインプレース方式でアップグレードする。
▼ 手順¶
(1)
-
CRDを更新する。
必要なCRDのマニフェストは、リポジトリで確認する必要がある。
$ git clone https://github.com/istio/istio.git
$ kubectl apply -f manifests/charts/base/crds
(2)
-
IstiodコントロールプレーンとIstio IngressGatewayの両方をインプレース方式でアップグレードする。
$ istioctl upgrade
(3)
-
データプレーンの
istio-proxy
コンテナを再インジェクションする。
$ kubectl rollout restart deployment app-deployment -n app
カナリア方式¶
▼ カナリア方式とは¶
▼ istioctl
コマンドの場合¶
▼ helm
コマンドの場合¶
(1)
-
HelmではCRDを管理しないようにし、
kubectl
コマンドでこれを作成する。
$ kubectl diff -f https://raw.githubusercontent.com/istio/istio/1.15.3/manifests/charts/base/crds/crd-all.gen.yaml
(2)
-
istiodチャートを使用して、古いバージョンのMutatingWebhookConfigurationのみを削除する。
この時、既存のリリースは古いリリースとして扱う。
$ helm upgrade <古いバージョンのリリース名> <チャートリポジトリ名>/istiod -n istio-system --version <古いバージョン> --set revisionTags=null
(3)
-
istiodチャートを使用して、新しいバージョンのMutatingWebhookConfigurationを作成しつつ、Istiodコントロールプレーンに関するKubernetesリソースを変更する。
この時、リリースを新しく命名する。
$ helm upgrade <新しいバージョンのリリース名> <チャートリポジトリ名>/istiod -n istio-system --version <新しいバージョン>
(4)
-
特定のNamespaceをアップグレードする。
(5)
-
動作確認し、問題なければ、残りのNamespaceもアップグレードする。
(6)
-
istiodチャートを使用して、古いリリースで作成したIstiodコントロールプレーンに関するKubernetesリソースを削除する。
$ helm upgrade <古いバージョンのリリース名> <チャートリポジトリ名>/istiod -n istio-system --version <古いバージョン> --set revisionTags=null
(7)
-
istio-baseを使用して、Istioに関するCRDを変更する。
この時、リリースを新しく命名する。
$ helm upgrade <新しいバージョンのリリース名> <チャートリポジトリ名>/base -n istio-system --version <新しいバージョン>
(8)
-
gatewayチャートを使用して、Istio IngressGatewayに関するKubernetesリソースを変更する。
この時、リリースを新しく命名する。
$ helm upgrade <新しいバージョンのリリース名> <チャートリポジトリ名>/gateway -n istio-ingress --version <新しいバージョン>
04. アーキテクチャ特性¶
性能¶
記入中...
05. CI¶
GitLab¶
▼ .gitlab-ci.yml
ファイル¶
CI上でClusterを作成し、Istioをデプロイする。
# ブランチ名に応じて、CIで使用する実行環境名を切り替える
workflow:
rules:
# masterブランチにて、任意の方法でパイプラインを実行した場合
- if: $CI_COMMIT_REF_NAME == 'master'
variables:
ENV: "prd"
# developブランチにて、任意の方法でパイプラインを実行した場合
- if: $CI_COMMIT_REF_NAME == 'develop'
variables:
ENV: "stg"
# MRにて、任意の方法でパイプラインを実行した場合
- if: $CI_PIPELINE_SOURCE == 'merge_request_event'
variables:
ENV: "tes"
# 上記以外で、webから手動でパイプラインを実行した場合
- if: $CI_PIPELINE_SOURCE == 'web'
variables:
ENV: "tes"
variables:
# EKSはK8sのマイナーバージョンを公開していないため、".0"と仮定して処理する
# 現在のEKSのK8sバージョン
K8S_CURRENT_VERSION: "1.24.0"
# アップグレード後のEKSのK8sバージョン
K8S_NEXT_VERSION: "1.26.0"
# 現在のIstioのバージョン
ISTIO_CURRENT_VERSION: "1.15.3"
# アップグレード後のIstioのバージョン
ISTIO_NEXT_VERSION: "1.17.5"
stages:
- build
- test
# K3Dの設定ファイルをセットアップする
setup_k3d_config:
stage: build
image: amazon/aws-cli
script:
- AWS_ECR="${AWS_ACCOUNT_ID}.dkr.ecr.${AWS_DEFAULT_REGION}.amazonaws.com"
- |
cat <<EOF > k3d-config.yaml
apiVersion: k3d.io/v1alpha5
kind: Simple
registries:
config: |
mirrors:
<AWSアカウントID>.dkr.ecr.ap-northeast-1.amazonaws.com:
endpoint:
- "http://<AWSアカウントID>.dkr.ecr.ap-northeast-1.amazonaws.com"
configs:
${AWS_ECR}:
auth:
username: AWS
password: $(aws ecr get-login-password --region ${AWS_DEFAULT_REGION})
EOF
# 指定したバージョンのIstioを検証する
test_istio:
stage: test
image:
name: docker
variables:
# K3Dを使用することで Docker in Docker となるため、そのための環境変数を設定する
# @see https://gitlab.com/gitlab-org/gitlab-runner/-/issues/27300
DOCKER_DRIVER: "overlay2"
DOCKER_HOST: "tcp://docker:2375"
DOCKER_TLS_CERTDIR: ""
services:
- name: docker:dind
command: ["--tls=false"]
# K3D Clusterは異なるJobに持ち越せないので、事前処理として実行する
before_script:
- apk --update add bash curl git
# スクリプトでasdfをセットアップする
- source setup-asdf.sh
# K3Dをインストールする
- |
curl -s https://raw.githubusercontent.com/k3d-io/k3d/main/install.sh | sh
k3d version
# kubectlコマンドをインストールする
- |
curl -kLO https://dl.k8s.io/release/v"${K8S_NEXT_VERSION}"/bin/linux/amd64/kubectl
chmod +x ./kubectl
mv ./kubectl /usr/local/bin/kubectl
kubectl version
# Clusterを作成する
# registries.yamlファイルをvolumeで配置する
# もし該当のバージョンのイメージがなければ、rc版を使用する
- |
if [ $? -ne 0 ]; then \
k3d cluster create --config k3d-config.yaml "${CI_PIPELINE_ID}" --image rancher/k3s:v"${K8S_NEXT_VERSION}"-k3s1 --agents 2; \
else \
k3d cluster create --config k3d-config.yaml "${CI_PIPELINE_ID}" --image rancher/k3s:v"${K8S_NEXT_VERSION}"-rc1-k3s1 --agents 2; \
fi
# Nodeにラベル付けする
- |
kubectl label node k3d-"${CI_PIPELINE_ID}"-agent-0 node.kubernetes.io/nodetype=ingress --overwrite
kubectl label node k3d-"${CI_PIPELINE_ID}"-agent-1 node.kubernetes.io/nodetype=system --overwrite
# 動作を確認する
- k3d cluster list
- kubectl get node --show-labels
# Istioのインストールは、Helmを使ったIstioのアップグレード手順に則る
# @see https://istio.io/latest/docs/setup/upgrade/helm/
script:
# CRDをインストールする
- kubectl apply -f https://raw.githubusercontent.com/istio/istio/"${ISTIO_NEXT_VERSION}"/manifests/charts/base/crds/crd-all.gen.yaml
# Namespaceを作成する
- |
kubectl create ns istio-ingress
kubectl label ns istio-ingress istio.io/rev=default
kubectl create ns istio-system
# Namespaceのラベルを確認する
- kubectl get ns -L istio.io/rev
# istiodチャートをApplyする
# ブルー/グリーンデプロイ時に新旧Istiodを並行稼働させるために、helmfile.yamlにリビジョン番号をつける
- helmfile -e "${ENV}" -f helmfile_istiod_"${ISTIO_NEXT_VERSION//\./-}".yaml apply --skip-crds --skip-diff-on-install
# istio-baseチャートをApplyする
- helmfile -e "${ENV}" -f helmfile_istio-base.yaml apply --skip-diff-on-install
# istio-ingressgatewayチャートをApplyする
- helmfile -e "${ENV}" -f helmfile_istio-ingressgateway.yaml apply --skip-diff-on-install
# 動作を確認する
- istioctl version
- istioctl proxy-status
- kubectl get all -n istio-ingress
- kubectl get all -n istio-system
# Clusterを削除する
- k3d cluster delete $CI_PIPELINE_ID
▼ setup-asdf.sh
ファイル¶
#!/bin/bash
# asdfをインストールする
git clone --depth 1 https://github.com/asdf-vm/asdf.git ~/.asdf
echo '. "$HOME/.asdf/asdf.sh"' >> ~/.bashrc
export ASDF_DIR=~/.asdf
source ~/.bashrc
# コマンドをインストールする
asdf plugin add helm https://github.com/Antiarchitect/asdf-helm.git
asdf plugin add helmfile https://github.com/feniix/asdf-helmfile.git
asdf plugin add istioctl https://github.com/virtualstaticvoid/asdf-istioctl.git
asdf install
asdf list
# 正しいバージョンをインストールできていることを確認する
helm version
helmfile version
istioctl version