#!/usr/bin/env bash # 3 options: # - docker compose # - docker "simple" # - classic installation # > macOS # > debian/ubuntu # > other linux config() { echo "detecting curl..." if ! command -v curl then echo "Please install curl before running this script." echo "curl was not found, exiting..." exit 1 fi echo "detecting jq..." if ! command -v jq then echo "Please install jq before running this script." echo "jq was not found, exiting..." exit 1 fi if ! command -v sudo then echo "Please install and configure sudo before running this script." echo "sudo was not found, exiting..." exit 1 fi echo "checking memory..." mem=$(free -mt | grep Total | awk '{print $2}') if [ "$mem" -lt 4000 ] then read -rp "Not enough memory to perform upgrade. Would you like to add the necessary swap? (y/n) " swap /dev/null | head -n 1 | cut -d$' ' -f2) if [ "$http_res" = "200" ] then echo "ONLINE" else echo "OFFLINE" fi } test_version() { local version=$(curl "$ES_IP:9200" 2>/dev/null | grep number | awk '{print $3}') case "$version" in *1.7*) echo "1.7" ;; *2.4*) echo "2.4" ;; *5.6*) echo "5.6" ;; *6.2*) echo "6.2" ;; *) echo "$version" esac } detect_installation() { echo "Detecting installation type..." test_docker_compose if [[ "$TYPE" = "DOCKER-COMPOSE" ]] then echo "Docker-compose installation detected." else test_docker if [[ "$TYPE" = "DOCKER" ]] then echo "Classical docker installation detected." else test_classic if [[ "$TYPE" = "CLASSIC" ]] then echo "Local installation detected on the host system." fi fi fi if [[ "$TYPE" = "NOT-FOUND" ]] then echo "ElasticSearch 1.7 was not found on the current system, exiting..." exit 2 else echo -n "Detecting online status... " if [[ "$TYPE" != "NOT-FOUND" ]] then STATUS=$(test_running) echo "$STATUS" if [ "$STATUS" = "OFFLINE" ] then echo "Your ElasticSearch instance is offline. Please check the logs and verify that no problems prevent the upgrade..." if [ "$TYPE" = "CLASSIC" ]; then echo "Note: You can use \`sudo journalctl -u elasticsearch.service\` or \`vi /var/log/elasticsearch/elasticsearch.log.$\(date --rfc-3339=date\)\` to view the logs, depending on your installation"; else echo "Note: You can use \`docker logs CONTAINER\` to view the logs"; fi fi fi fi } error_index_status() { echo "Your elasticSearch installation contains indices which states are not \"green\", but this cannot be solved automatically." echo "Please solve theses status before continuing..." curl "$ES_IP:9200/_cat/indices?v" 2>/dev/null | grep --color -E "yellow|red|$" exit 6 } ensure_initial_status_green() { echo "Checking status of your elasticSearch indices..." local state=$(curl "$ES_IP:9200/_cat/indices" 2>/dev/null | awk '{print $1}' | sort | uniq) if [ "$state" != "green" ] then local replicas=$(curl "$ES_IP:9200/_cat/indices" 2>/dev/null | awk '{print $5}' | sort | uniq) if [ "$replicas" != "0" ] then local indices=$(curl "$ES_IP:9200/_cat/indices" 2>/dev/null | awk '{print $3}') for index in $indices # do not surround $indices with quotes do curl -XPUT "$ES_IP:9200/$index/_settings" 2>/dev/null -H 'Content-Type: application/json' -d '{ "index": { "number_of_replicas": 0 } }' done local final_state=$(curl "$ES_IP:9200/_cat/indices" 2>/dev/null | awk '{print $1}' | sort | uniq) if [ "$final_state" != "green" ]; then error_index_status; fi else error_index_status fi fi } wait_for_online() { local counter=0 echo -n "Waiting for ElasticSearch instance to came online" STATUS=$(test_running) while [ "$STATUS" != "ONLINE" ] do echo -n "." sleep 1 STATUS=$(test_running) ((++counter)) if [ "$counter" -eq 120 ] then echo -e "\nThe ElasticSearch instance did not came online for 2 minutes, please check the logs for any errors. Exiting..." exit 8 fi done echo -e "\n" } wait_for_green_status() { echo -n "Waiting for ElasticSearch indices to have green status" local state=$(curl "$ES_IP:9200/_cat/indices" 2>/dev/null | awk '{print $1}' | sort | uniq) while [ "$state" != "green" ] do echo -n "." sleep 1 state=$(curl "$ES_IP:9200/_cat/indices" 2>/dev/null | awk '{print $1}' | sort | uniq) done echo -e "\n" } prepare_upgrade() { curl -XPUT "$ES_IP:9200/_cluster/settings?pretty" 2>/dev/null -H 'Content-Type: application/json' -d'{ "transient": { "cluster.routing.allocation.enable": "none" } }' curl -XPOST 2>/dev/null "$ES_IP:9200/_flush/synced?pretty" } reenable_allocation() { curl -XPUT "$ES_IP:9200/_cluster/settings?pretty" -H 'Content-Type: application/json' -d'{ "transient": { "cluster.routing.allocation.enable": "all" } }' } upgrade_compose() { local current=$1 local target=$2 echo -e "\nUpgrading docker-compose installation from $current to $target..." prepare_upgrade docker-compose stop elasticsearch docker-compose rm -f elasticsearch local image="elasticsearch:$target" if [ $target = '6.2' ]; then image="docker.elastic.co\/elasticsearch\/elasticsearch-oss:6.2.3"; fi sed -i.bak "s/image: elasticsearch:$current/image: $image/g" "$FM_PATH/docker-compose.yml" if ! grep -qe "ES_JAVA_OPTS" docker-compose.yml then sed -i.bak "/image: $image/s/.*/&\n environment:\n - \"ES_JAVA_OPTS=-Xms512m -Xmx512m\"/" "$FM_PATH/docker-compose.yml" fi if ! grep -qe "ulimits" docker-compose.yml then sed -i.bak "/image: $image/s/.*/&\n ulimits:\n memlock:\n soft: -1\n hard: -1/" "$FM_PATH/docker-compose.yml" fi docker-compose pull docker-compose up -d wait_for_online wait_for_green_status # check status local version=$(test_version) if [ "$STATUS" = "ONLINE" ] && [ "$version" = "$target" ]; then echo "Installation of elasticsearch $target was successful." else echo "Unable to find an active ElasticSearch $target instance, something may have went wrong, exiting..." echo "status: $STATUS, version: $version" exit 4 fi } upgrade_docker() { local current=$1 local target=$2 echo -e "\nUpgrading docker installation from $current to $target..." local containers=$(docker ps | grep "elasticsearch:$current") # get container id local id=$(docker inspect -f '{{.Id}} {{range .NetworkSettings.Networks}}{{.IPAddress}}{{end}}' $(echo "$containers" | awk '{print $1}') | grep "$ES_IP" | awk '{print $1}') # get container name local name=$(docker inspect -f '{{.Name}}' "$id" | sed s:^/::g) # get container network name local network=$(docker inspect -f '{{.NetworkSettings.Networks}}' "$id" | sed 's/map\[\(.*\):0x[a-f0-9]*\]/\1/') # get container mapping to data folder local mounts=$(docker inspect -f '{{.Mounts}}' "$id" | sed 's/} {/\n/g' | sed 's/^\[\?{\?bind[[:blank:]]*\([^[:blank:]]*\)[[:blank:]]*\([^[:blank:]]*\)[[:blank:]]*true \(rprivate\)\?}\?]\?$/-v \1:\2/g' | sed -e ':a' -e 'N' -e '$!ba' -e 's/\n/ /g') prepare_upgrade # stop current elastic docker stop "$name" docker rm -f "$name" # run target elastic local image="elasticsearch:$target" local image_name="$image" if [ $target = '6.2' ] then image_name="elasticsearch-oss:$target" image="docker.elastic.co/elasticsearch/$image_name" fi docker pull "$image" echo docker run --restart=always -d --name="$name" --network="$network" --ip="$ES_IP" "$mounts" "$image_name" | bash wait_for_online wait_for_green_status # check status local version=$(test_version) if [ "$STATUS" = "ONLINE" ] && [ "$version" = "$target" ]; then echo "Installation of elasticsearch $target was successful." else echo "Unable to find an active ElasticSearch $target instance, something may have went wrong, exiting..." echo "status: $STATUS, version: $version" exit 4 fi } unsupported_message() { local version=$1 echo "Automated upgrade of your elasticSearch installation is not supported on your system." echo "Please refer to your vendor's instructions to install ElasticSearch $version" echo "For more informations: https://www.elastic.co/guide/en/elasticsearch/reference/$version/setup-upgrade.html" exit 5 } upgrade_classic() { local target=$1 local system=$(uname -s) case "$system" in Linux*) if [ -f /etc/os-release ] then . /etc/os-release if [ "$ID" = 'debian' ] || [[ "$ID_LIKE" = *'debian'* ]] then # Debian compatible echo -e "\nUpdating ElasticSearch to $target" prepare_upgrade # stop elasticsearch if command -v systemctl then sudo systemctl stop elasticsearch.service else sudo /etc/init.d/elasticsearch stop fi # process with package upgrade wget -qO - https://packages.elastic.co/GPG-KEY-elasticsearch | sudo apt-key add - case "$target" in "2.4") echo "deb http://packages.elastic.co/elasticsearch/2.x/debian stable main" | sudo tee /etc/apt/sources.list.d/elasticsearch-2.x.list ;; "5.6") echo "deb https://artifacts.elastic.co/packages/5.x/apt stable main" | sudo tee /etc/apt/sources.list.d/elastic-5.x.list ;; "6.2") echo "deb https://artifacts.elastic.co/packages/6.x/apt stable main" | sudo tee /etc/apt/sources.list.d/elastic-6.x.list ;; esac sudo apt-get update && sudo apt-get install --only-upgrade elasticsearch # restart elasticsearch service if command -v systemctl then sudo systemctl start elasticsearch.service else sudo /etc/init.d/elasticsearch start fi else unsupported_message fi fi ;; Darwin*) # Mac OS X echo -e "\nUpdating ElasticSearch to $target" prepare_upgrade brew services stop elasticsearch brew update case "$target" in "2.4") brew install elasticsearch@2.4 ;; "5.6") brew install elasticsearch@5.6 ;; "6.2") brew install elasticsearch ;; esac brew services start elasticsearch ;; *) unsupported_message ;; esac wait_for_online wait_for_green_status # check status local version=$(test_version) if [ "$STATUS" = "ONLINE" ] && [ "$version" = "$target" ]; then echo "Installation of elasticsearch $target was successful." else echo "Unable to find an active ElasticSearch $target instance, something may have went wrong, exiting..." echo "status: $STATUS, version: $version" exit 4 fi } reindex_indices() { # get number of documents (before elastic 5.x, docs.count is at column 6) local docs=$(curl "$ES_IP:9200/_cat/indices" 2>/dev/null | awk '{s+=$6} END {printf "%.0f", s}') # get all indices local indices=$(curl "$ES_IP:9200/_cat/indices" 2>/dev/null | awk '{print $3}') local migration_indices="" for index in $indices # do not surround $indices with quotes do # get the mapping of the existing index local mapping=$(curl "http://$ES_IP:9200/$index/_mapping" 2>/dev/null | jq -c -M -r ".$index") local definition=$(echo "$mapping" '{ "settings" : { "index" : { "number_of_shards": 1, "number_of_replicas": 0, "refresh_interval": -1 } } }' | jq -s add) local migration_index="$index""_$1" migration_indices+="$migration_index," # create the temporary migration index with the previous mapping curl -XPUT "http://$ES_IP:9200/$migration_index/" 2>/dev/null -H 'Content-Type: application/json' -d "$definition" # reindex data content to the new migration index curl -XPOST "$ES_IP:9200/_reindex?pretty" 2>/dev/null -H 'Content-Type: application/json' -d '{ "source": { "index": "'"$index"'" }, "dest": { "index": "'"$migration_index"'" } }' done echo "Indices are reindexing. This may take a while, waiting to finish... " # first we wait for all indices states became green wait_for_green_status # then we wait for all documents to be reindexed local new_docs=$(curl "$ES_IP:9200/_cat/indices?index=$migration_indices" 2>/dev/null | awk '{s+=$6} END {printf "%.0f", s}') while [ "$new_docs" != "$docs" ] do echo -ne "\rdocs: $docs, reindexed: $new_docs" sleep 1 new_docs=$(curl "$ES_IP:9200/_cat/indices?index=$migration_indices" 2>/dev/null | awk '{s+=$6} END {printf "%.0f", s}') done echo -e "\nReindex completed, deleting previous index..." for index in $indices # do not surround $indices with quotes do curl -XDELETE "$ES_IP:9200/$index?pretty" 2>/dev/null done reenable_allocation } reindex_final_indices() { local previous=$1 # get number of documents (from elastic 5.x, docs.count is at column 7) local docs=$(curl "$ES_IP:9200/_cat/indices" 2>/dev/null | awk '{s+=$7} END {printf "%.0f", s}') # get all indices local indices=$(curl "$ES_IP:9200/_cat/indices" 2>/dev/null | awk '{print $3}') local final_indices="" for index in $indices # do not surround $indices with quotes do # get the mapping of the existing index local mapping=$(curl "http://$ES_IP:9200/$index/_mapping" 2>/dev/null | jq -c -M -r ".$index") local definition=$(echo "$mapping" '{ "settings" : { "index" : { "number_of_shards": 1, "number_of_replicas": 0, "refresh_interval": -1 } } }' | jq -s add) local final_index=$(echo "$index" | sed "s/\(.*\)_$previous$/\1/") final_indices+="$final_index," # create the final index with the previous mapping curl -XPUT "http://$ES_IP:9200/$final_index" 2>/dev/null -H 'Content-Type: application/json' -d "$definition" # reindex data content to the new migration index curl -XPOST "$ES_IP:9200/_reindex?pretty" 2>/dev/null -H 'Content-Type: application/json' -d '{ "source": { "index": "'"$index"'" }, "dest": { "index": "'"$final_index"'" } }' done echo "Indices are reindexing. This may take a while, waiting to finish... " # first we wait for all indices states became green wait_for_green_status # then we wait for all documents to be reindexed local new_docs=$(curl "$ES_IP:9200/_cat/indices?index=$final_indices" 2>/dev/null | awk '{s+=$7} END {printf "%.0f", s}') while [ "$new_docs" != "$docs" ] do echo -ne "\rdocs: $docs, reindexed: $new_docs" sleep 1 new_docs=$(curl "$ES_IP:9200/_cat/indices?index=$final_indices" 2>/dev/null | awk '{s+=$7} END {printf "%.0f", s}') done echo -e "\nReindex completed, deleting previous index..." for index in $indices # do not surround $indices with quotes do curl -XDELETE "$ES_IP:9200/$index?pretty" 2>/dev/null done reenable_allocation } start_upgrade() { # $1: current version # $2: target version case "$TYPE" in "DOCKER-COMPOSE") upgrade_compose $1 $2 ;; "DOCKER") upgrade_docker $1 $2 ;; "CLASSIC") upgrade_classic $2 ;; *) echo "Unexpected ElasticSearch installation $TYPE" exit 3 esac } upgrade_elastic() { config detect_installation read -rp "Continue with upgrading? (y/n) " confirm