This project is migrated from:https://github.com/bindiego/local_services/tree/develop/k8s/gke/elastic
Known production environment: 40 all role nodes evenly spread over 2 zones with 200TB data pd mounted.
Suppose you already have access to Google Cloud Platform with proper permissions. We will use Official ECK operator with Official Containers / Dockers & Docker source repo. They are all open source and free, especially the operator can handle the Elasticsearch nodes upgrades/migrations in a graceful way, and lot more.
Once you checked out this repo, make sure you stay in this folder as your working directory, local_services/k8s/gke/elastic
In case you do not have gcloud
installed, you can run ./bin/gcloud.sh install
to get it. This will have the cloud SDK installed in your $HOME/google-cloud-sdk
directory. Add the full path of bin
folder to your $PATH
to make it works.
Run ./bin/gcloud.sh
for other usages, but importantly, make sure you have kubectl
properly installed, or you can run ./bin/gcloud.sh kubectl
to have it setted up.
Now we are good to go!
By default, the GKE resources will be launched in us-central1
, a.k.a Iowa, USA, region as a zonal cluster in zone-a to minimize your demo/PoC costs. Feel free to change & choose one close to you.
After that you are all set.
./bin/demo.sh
will setup everything for you. Once completed, you should seen something similar to the following output
=================================
Elasticsearch status:
{
"name" : "dingo-demo-es-zone-a-2",
"cluster_name" : "dingo-demo",
"cluster_uuid" : "cnjICb9WRdCNkIhxsMYM8w",
"version" : {
"number" : "8.15.5",
"build_flavor" : "default",
"build_type" : "docker",
"build_hash" : "b10896bcfe167cce44a84ba2771d101fb596d40d",
"build_date" : "2024-11-21T22:06:13.985834967Z",
"build_snapshot" : false,
"lucene_version" : "9.11.1",
"minimum_wire_compatibility_version" : "7.17.0",
"minimum_index_compatibility_version" : "7.0.0"
},
"tagline" : "You Know, for Search"
}
---------------------------------
Kibana: https://35.201.182.178:5601
Elasticsearch: https://35.234.3.182:9200
Username: elastic
Password: password
=================================
and you could always retrieve above information by running ./bin/demo.sh status
如果最后的output里没有正确输出连接信息,可以等待几分钟时间再次尝试手动查询集群链接状态 ./bin/demo.sh status
./bin/demo.sh 6.8
will launch an old 6.8.23
Elastic Stack which is the oldest version we suport by now.
Check the status by running the following command
passwd=$(kubectl get secret dingo-demo-6-es-elastic-user -o go-template='{{.data.elastic | base64decode}}')
lb_ip=`kubectl get services dingo-demo-6-es-http -o jsonpath='{.status.loadBalancer.ingress[0].ip}'`
kbn_ip=`kubectl get service dingo-demo-kbn-6-kb-http -o jsonpath='{.status.loadBalancer.ingress[0].ip}'`
kbn_url=https://${kbn_ip}:5601
curl -u "elastic:$passwd" -k "https://$lb_ip:9200"
echo "Kibana: " ${kbn_url}
echo "Elasticsearch: " "https://$lb_ip:9200"
echo "Username: " elastic
echo "Password: " ${passwd}
echo "================================="; echo
To tidy things up and release all the created cloud resources, simply run ./bin/demo.sh clean
Check the Advanced topics if you would like to:
- Customize the size of your k8s/GKE nodes and pools
- Elasticsearch topology
- k8s/GKE & Elasticsearch node sizing
- Use your own meaningful names for workloads/services/ingress etc.
Alternatives to start the deployment
OK, let's get down to the business.
Locate the project ID. Open the GCP console, it's on the top-left corner "Project info" card by default.
Now, replace the project_id
variable in ./bin/gke.sh with your own project id.
For targeting region, you will need to update two files
Change the region
variable on your choice, us-central1
by default.
You can later adjust all these settings to archieve your own goal. We will discuss more in Advanced topics.
Run make init_single
and you done.
zone-a | zone-b |
---|---|
ES x 2 | ES x 2 |
Run make init_allrole
Option 3: Production deployments, with shard allocation awareness (not forced) 角色分离部署(适合中大规模集群,最好针对搜索或者分析场景进行集群分离和相关内存的和IO的配置调优)
zone-a | zone-b |
---|---|
Master x 2 | Master x 1 |
Ingest x 1 | Ingest x 1 |
Data x 2 | Data x 2 |
Coordinating x 1 | Coordinating x 1 |
ML x 1 | on any available node |
You could even further adjust this to a single zone or 3 zones with forced shared allocation awareness, let's discuss more details in the Advanced topics so you could configure based on your needs.
Run make init_prod
We use this credential for Elasticsearch cluster to manage it's snapshots. On GKE, we use GCS (an object store on GCP) for that purpose.
Please consult How to create & manage service accounts. Simply download the key, a json file, then name it gcs.client.default.credentials_file
and put it into conf
dir. We only need the permission to manipulate GCS, so make sure it has only the minimum permissions granted.
or Run ./bin/gcs_serviceaccount.sh
to create a service account and generate the json file to ./conf/gcs.client.default.credentials_file
. Please change the varialbes in the ./bin/gcs_serviceaccount.sh
for your environment.
By now, in your working directory, you should be able to run cat ./conf/gcs.client.default.credentials_file
to check the existence and the contents of the file. If you didn't do this, the auto script will later use $GOOGLE_APPLICATION_CREDENTIALS
environment variable to copy that file to the destination. You cannot skip this by now let's talk about how to disble in Advanced topics if you really have to.
Run ./bin/gke.sh create
You can easily change whatever you want, but you'd better think about this by now since you may need to update the deployment files.
We enabled all data security options, so all the communications are secured too.
The only prerequisite here is you need a domain name, let's assume it is bindiego.com
.
Let's take an overview of the connections,
Data ---https---> Global Load Balancer ---https---> Ingest nodes
Clients ---https---> Global Load Balancer ---https---> Coordinating nodes
Kibana --https---> Global Load Balancer ---https---> Kibana(s)
As you can see, we don't terminate the https
protocol for the LB's backends. You can certainly do by updating the yml files accordingly, let's talk about that later.
It's the time to reserve an Internet IP address for your domain.
Run ./bin/glb.sh reserve
, this should print out the actual IP address for you. If you missed for whatever reason, you will always be able to retrieve it by ./bin/glb.sh status
OK, time to configure your DNS
I have added 3 sub-domains for Kibana, Elasticsearch ingest & Elasticsearch client respectively. You can do that by adding 3 A records
- k8na.bindiego.com
- k8es.ingest.bindiego.com
- k8es.client.bindiego.com
all pointing to the static IP address you just reserved.
Optionally, you may want to add a DNS CAA record to specify the authority who sign the certificate. This is super important for China mainland users, since in this quickstart sample we are going to use Google managed certificate for simplicity. Or Google signed ones will have the keyword Google and potentially could be blocked by the GFW.
- Google
CAA 0 issue "pki.goog"
- Letsencrypt
CAA 0 issue "letsencrypt.org"
You will need to update 2 files now,
Better do a quick find & replace
in the text editor of your choice to have those domains configured properly according to your environment.
Once you done, it's the time to run ./bin/glb.sh cert
, wait the last step to deploy the GLB.
This one is really simple, depends on which service you would like to expose, simply uncomment the spec.http
sections in either ./deploy/es.yml
or ./deploy/kbn.yml
or both. And you do not need to deploy the GLB in the end as you will do for option 1.
This will setup up regional TCP LB for your deployments respectively. Make sure you access the ip:port
by using https
protocol.
Be cautious if you didn't setup the certificates properly or used the default custom ones, when you try to connect to this secured (ssl enabled) cluster. You could simply bypass the verification in curl by using curl --insecure
option. But for encrypted communication in code, here let's use java as an example, you could consult the official docs
这里要非常注意的就是如果你自己生成的证书,或者使用默认的证书,你应该会遇到下面的问题,我们这里提供了解决方法。
As you could see in the sample code, you will need to craft an SSLContext
, something like this:
try {
// SSLContext context = SSLContext.getInstance("SSL");
SSLContext context = SSLContext.getInstance("TLS");
context.init(null, new TrustManager[] {
new X509TrustManager() {
public void checkClientTrusted(X509Certificate[] chain, String authType) {}
public void checkServerTrusted(X509Certificate[] chain, String authType) {}
public X509Certificate[] getAcceptedIssuers() { return null; }
}
}, null);
httpAsyncClientBuilder.setSSLContext(context)
.setSSLHostnameVerifier(NoopHostnameVerifier.INSTANCE);
} catch (NoSuchAlgorithmException ex) {
logger.error("Error when setup dummy SSLContext", ex);
} catch (KeyManagementException ex) {
logger.error("Error when setup dummy SSLContext", ex);
} catch (Exception ex) {
logger.error("Error when setup dummy SSLContext", ex);
}
Basically, the above code does two things
- A Dummy SSL context
- Turn off the Hostname verifier
./bin/es.sh deploy
Services may take a little while (1min or 2) to in a ready status. Check it in the GCP console or by kubectl
commands.
./bin/es.sh password
You will need this credential to login to Kibana or talking to Elasticsearch cluster. For the latest version, when interacting with Elasticsearch, you may want a token based authentication rather than embeded usernname & password.
./bin/kbn.sh deploy
Or manually edit the Health Check, replace /
with /login
after you deployed the ingress.
./bin/glb.sh deploy
After this you should be able to test your cluster, assume the domain is bindiego.com
-
Access Kibana via
k8na.bindiego.com
with usernameelastic
and password from./bin/es.sh password
-
Test Elasticsearch ingest nodes
curl -u "elastic:<passwd>" -k "https://k8es.ingest.bindiego.com"
- Test Elasticsearch coordinating nodes
curl -u "elastic:<passwd>" -k "https://k8es.client.bindiego.com"
Feel free to remove -k
option since your certificate is managed by Google.
We done by now. Further things todo:
- Setup the file repo & snapshots to backup your data
- Go to Advanced topics for more complex setups, mostly about manipulating yml files.
Configure GCS (Google Cloud Storage) bucket
PUT /_snapshot/dingo_gcs_repo
{
"type": "gcs",
"settings": {
"bucket": "bucket_name",
"base_path": "es_backup",
"compress": true
}
}
or
curl -X PUT \
-u "elastic:<password>" \
"https://k8es.client.bindiego.com/_snapshot/dingo_gcs_repo" \
-H "Content-Type: application/json" -d '{
"type": "gcs",
"settings": {
"bucket": "bucket_name",
"base_path": "es_backup",
"compress": true
}
}'
Test snapshot
PUT /_snapshot/dingo_gcs_repo/test-snapshot
or, take a daily snapshot with date as part of the name
curl -X PUT \
-u "elastic:<password>" \
"https://k8es.client.bindiego.com/_snapshot/dingo_gcs_repo/test-snapshot_`date +'%Y_%m_%d'`" \
-H "Content-Type: application/json" -d '{}'
More details about Snapshot & restore and lifecycle policies etc.
kubectl create secret generic kibana-elasticsearch-credentials --from-literal=elasticsearch.password=$PASSWORD
apiVersion: kibana.k8s.elastic.co/v1
kind: Kibana
metadata:
name: kbn
spec:
version: 8.15.5
count: 1
config:
elasticsearch.hosts:
- https://elasticsearch.example.com:9200
elasticsearch.username: elastic
secureSettings:
- secretName: kibana-elasticsearch-credentials
or the Elasticsearch cluster is using a slef-signed certificate, create a k8s secret containing the CA certificate and mount to the Kibana container as follows
apiVersion: kibana.k8s.elastic.co/v1
kind: Kibana
metadata:
name: kbn
spec:
version: 8.15.5
count: 1
config:
elasticsearch.hosts:
- https://elasticsearch-sample-es-http:9200
elasticsearch.username: elastic
elasticsearch.ssl.certificateAuthorities: /etc/certs/ca.crt
secureSettings:
- secretName: kibana-elasticsearch-credentials
podTemplate:
spec:
volumes:
- name: elasticsearch-certs
secret:
secretName: elasticsearch-certs-secret
containers:
- name: kibana
volumeMounts:
- name: elasticsearch-certs
mountPath: /etc/certs
readOnly: true
Use the ./bin/apm.sh
script for deploy
, clean
, password
./bin/es.sh pw
./bin/es.sh pwreset
We have predefined 4 different types of storage, you could refer to the ./deploy/es.yml
file, section spec.nodeSets[].volumeClaimTemplates.spec.storageClassName
to find out what we used for each different ES node.
可以根据上面的链接各个磁盘的性能,对不同角色的节点选取相应的存储来实现资源的最佳利用。下面针对每个存储也给出了一些适合的建议。
- dingo-pdssd 高性能
type: zonal SSD
best for: Data nodes (hot/warm), Master nodes, ML nodes
- dingo-pdssd-ha 高性能高可用
type: regional SSD
best for: Master nodes, Data nodes (hot/warm)
- dingo-pdssd-balanced 中等性能
type: zonal balanced SSD
best for: Data nodes (warm/cold), Master nodes, ML nodes
- dingo-pdssd-blanced-ha 中等性能高可用
type: regional balanced SSD
best for: Master nodes, Data nodes (warm/cold)
- dingo-pdhdd 磁盘
type: zonal HDD
best for: ML nodes, Ingest nodes, Coordinating nodes, Kibana, APM, Data nodes (cold)
- dingo-pdhdd-ha 高可用磁盘
type: regional HDD
best for: Data nodes (cold)
Topology is defined by spec.nodeSets
, you may adjust accordingly to your design apart from the predefined ones
Affnity is controlled in deployment yaml, used to do shard allocation awareness etc.
GKE specific node pool selection is configured in deployment yaml, this could speparate different roles with different k8s/GKE nodes. E.g. Kibana, APM server and various Elasticsearch nodes.
This could be a long topic to discuss, we actually want to hide the size of GKE node under the hood, let it depends on the Elasticsearch node's size. Let's skip the details and directly draw the conclusion,
- We have set the ES nodeSet affinity so data nodes will try to avoid host on the same machine (VM)
- If we can limit the GKE node size slightly larger than ES node, then we may avoid sharing compute resources across different nodeSets / roles
- For all role ES nodes, Pod Topology Spread Constraints is another way to evenly ditribute ES nodes rather than set
affinity
. In that way we may not be able to do the shared allocation awarenness by using current version of ECK.
./bin/gke.sh scale <number>
NOTE: the <nubmer>
here is number of nodes in each zone
- Elasticsearch
Update spec.nodeSets.count
for the specific group of nodes, then ./bin/es.sh deploy
- All others
Update spec.count
, then ./bin/kbn.sh deploy
for Kibana and so on so forth.
- Elasticsearch
Node memory: spec.nodeSets.podTemplate.spec.containers.resources.requests.memory
& spec.nodeSets.podTemplate.spec.containers.resources.limits.memory
JVM heap size: spec.nodeSets.podTemplate.spec.containers.env
for variable ES_JAVA_OPTS
We generally double the total memory upon JVM heap for Data nodes
with 64GB maximum. Here is the reason and why you should keep the heap size between 26GB and 31GB.
Other nodes we only add 1GB extra above the heap size, hence uaually 32GB maxinum. Very occasionally, you may need your coordinating nodes beyond that, consult our ES experts you could reach out :)
- All others
Memory: spec.podTemplate.spec.containers.resources.requests.memory
& spec.podTemplate.spec.containers.resources.limits.memory
Simply update spec.version
in yaml and make <your choice of topology>
, then run ./deploy/es.sh deploy
and you done. All other services, e.g. Kibana, APM are the same.
只要把部署yaml文件里的版本升级到目标版本,再次 deploy 就可以了。整个升级动作会由operator自动操作完成,过程根据情况可能很快也会比较漫长。
NOTE: downgrade is NOT supported 降级是不支持的
We have always set spec.nodeSets.updateStrategy.changeBudget.maxUnavailable
smaller than spec.nodeSets.count
, usually N - 1
. If the count
is 1
, then set the maxUnavailable
to -1
.
In case if you have 3 master nodes across 3 zones and defined in 3 nodeSets, you do not have to worry about they may offline at the same time. The ECK operator could handle that very well :)
- In case the pod is stucking in "Terminating" status, you could force delete it. Use command
kubectl get pod
to check the status and name of the pod. Then delete withkubectl delete pods <pod> --grace-period=0 --force
. More details.
This will trigger a rolling restart on all managed pods.
git pull
./bin/upgrade_ECK.sh
Cautious: deleteing the unassigned shard is the one last bet to fix your cluster health. You should always try to reaasign or recover the data instead. Here lets only focus on deletion.
- check unassigned shards
curl --user elastic:<passwod> -XGET http://<elastichost>:9200/_cluster/health?pretty | grep unassigned_shards
- retriev unassigned shards
curl -XGET http://<elastichost>:9200/_cat/shards | grep UNASSIGNED | awk {'print $1'}
- do it :)
curl -XGET http://<elastichost>:9200/_cat/shards | grep UNASSIGNED | awk {'print $1'} | xargs -i curl -XDELETE "http://<elastichost>:9200/{}"