2020-06-30 16:11:12 +02:00
|
|
|
|
#!/usr/bin/env bash
|
|
|
|
|
|
|
|
|
|
parseparams()
|
|
|
|
|
{
|
|
|
|
|
COMMANDS=()
|
|
|
|
|
SCRIPTS=()
|
|
|
|
|
ENVIRONMENTS=()
|
2021-06-01 08:54:19 +02:00
|
|
|
|
PREPROCESSING=()
|
2021-06-11 13:47:25 +02:00
|
|
|
|
while getopts "hyt:s:p:c:e:" opt; do
|
2020-06-30 16:11:12 +02:00
|
|
|
|
case "${opt}" in
|
|
|
|
|
y)
|
|
|
|
|
Y=true
|
|
|
|
|
;;
|
2021-06-11 13:47:25 +02:00
|
|
|
|
t)
|
|
|
|
|
TARGET=$OPTARG
|
|
|
|
|
FORCE_TARGET=true
|
|
|
|
|
;;
|
2020-06-30 16:11:12 +02:00
|
|
|
|
s)
|
|
|
|
|
SCRIPTS+=("$OPTARG")
|
|
|
|
|
;;
|
2021-06-01 08:54:19 +02:00
|
|
|
|
p)
|
|
|
|
|
PREPROCESSING+=("$OPTARG")
|
|
|
|
|
;;
|
2020-06-30 16:11:12 +02:00
|
|
|
|
c)
|
|
|
|
|
COMMANDS+=("$OPTARG")
|
|
|
|
|
;;
|
|
|
|
|
e)
|
|
|
|
|
ENVIRONMENTS+=("$OPTARG")
|
|
|
|
|
;;
|
|
|
|
|
*)
|
|
|
|
|
usage
|
|
|
|
|
;;
|
|
|
|
|
esac
|
|
|
|
|
done
|
|
|
|
|
shift $((OPTIND-1))
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
yq() {
|
2021-03-01 10:13:48 +01:00
|
|
|
|
docker run --rm -i -v "${PWD}:/workdir" mikefarah/yq:4 "$@"
|
2020-06-30 16:11:12 +02:00
|
|
|
|
}
|
|
|
|
|
|
2020-10-21 13:13:13 +02:00
|
|
|
|
jq() {
|
|
|
|
|
docker run --rm -i -v "${PWD}:/data" imega/jq "$@"
|
|
|
|
|
}
|
|
|
|
|
|
2021-06-11 13:47:25 +02:00
|
|
|
|
# set $SERVICE and $YES_ALL
|
2020-06-30 16:11:12 +02:00
|
|
|
|
config()
|
|
|
|
|
{
|
2021-06-11 13:47:25 +02:00
|
|
|
|
echo -e "Checking user... "
|
2020-06-30 16:11:12 +02:00
|
|
|
|
if [[ "$(whoami)" != "root" ]] && ! groups | grep docker
|
|
|
|
|
then
|
|
|
|
|
echo "Please add your current user to the docker group OR run this script as root."
|
|
|
|
|
echo "current user is not allowed to use docker, exiting..."
|
|
|
|
|
exit 1
|
|
|
|
|
fi
|
2021-04-19 17:07:03 +02:00
|
|
|
|
echo -e "Checking installation..."
|
|
|
|
|
if [ ! -f "docker-compose.yml" ]; then
|
|
|
|
|
echo -e "\e[91m[ ❌ ] docker-compose.yml was not found in ${PWD}. Please run this script from the Fab-manager's installation folder. Exiting... \e[39m"
|
|
|
|
|
exit 1
|
|
|
|
|
fi
|
2020-06-30 16:11:12 +02:00
|
|
|
|
|
2021-03-01 10:13:48 +01:00
|
|
|
|
SERVICE="$(yq eval '.services.*.image | select(. == "sleede/fab-manager*") | path | .[-2]' docker-compose.yml)"
|
2020-06-30 16:11:12 +02:00
|
|
|
|
YES_ALL=${Y:-false}
|
|
|
|
|
# COMMANDS, SCRIPTS and ENVIRONMENTS are set by parseparams
|
2021-04-19 17:07:03 +02:00
|
|
|
|
|
|
|
|
|
if [ -z "${SERVICE}" ]; then
|
|
|
|
|
echo -e "\e[91m[ ❌ ] The service name was not determined. Please check your docker-compose.yml file. Exiting... \e[39m"
|
|
|
|
|
exit 1
|
|
|
|
|
fi
|
2021-05-18 09:40:35 +02:00
|
|
|
|
echo -e "\n"
|
2020-06-30 16:11:12 +02:00
|
|
|
|
}
|
|
|
|
|
|
2020-10-06 14:18:35 +02:00
|
|
|
|
# compare versions utilities
|
|
|
|
|
# https://stackoverflow.com/a/4024263/1039377
|
|
|
|
|
verlte() {
|
|
|
|
|
[ "$1" = "$(echo -e "$1\n$2" | sort -V | head -n1)" ]
|
|
|
|
|
}
|
|
|
|
|
verlt() {
|
|
|
|
|
[ "$1" = "$2" ] && return 1 || verlte "$1" "$2"
|
|
|
|
|
}
|
|
|
|
|
|
2021-06-11 13:47:25 +02:00
|
|
|
|
# set $TAG and $TARGET
|
|
|
|
|
target_version()
|
|
|
|
|
{
|
|
|
|
|
TAG=$(yq eval ".services.$SERVICE.image" docker-compose.yml | grep -o ':.*')
|
|
|
|
|
|
|
|
|
|
if [ -n "$TARGET" ]; then return; fi
|
|
|
|
|
|
|
|
|
|
if [[ "$TAG" =~ ^:release-v[\.0-9]+$ ]]; then
|
|
|
|
|
TARGET=$(echo "$TAG" | grep -Eo '[\.0-9]{5}')
|
|
|
|
|
elif [ "$TAG" = ":latest" ]; then
|
|
|
|
|
TARGET=$(\curl -sSL "https://hub.fab-manager.com/api/versions/latest" | jq -r '.semver')
|
|
|
|
|
else
|
|
|
|
|
TARGET='custom'
|
|
|
|
|
fi
|
|
|
|
|
}
|
|
|
|
|
|
2020-10-06 14:18:35 +02:00
|
|
|
|
version_error()
|
|
|
|
|
{
|
2021-04-19 17:07:03 +02:00
|
|
|
|
printf "\n\n\e[91m[ ❌ ] You are running Fab-manager version %s\n\e[39m" "${VERSION:-undetermined}"
|
2021-06-11 13:47:25 +02:00
|
|
|
|
printf "You must upgrade Fab-manager to %s.\nPlease refer to http://update.doc.fab.mn for instructions\n" "$1" 1>&2
|
2020-10-06 14:18:35 +02:00
|
|
|
|
exit 3
|
|
|
|
|
}
|
|
|
|
|
|
2021-06-11 13:47:25 +02:00
|
|
|
|
# set $VERSION
|
2020-10-06 14:18:35 +02:00
|
|
|
|
version_check()
|
|
|
|
|
{
|
2021-03-09 09:25:27 +01:00
|
|
|
|
VERSION=$(docker-compose exec -T "$SERVICE" cat .fabmanager-version 2>/dev/null)
|
2020-10-06 14:18:35 +02:00
|
|
|
|
if [[ $? = 1 ]]; then
|
2021-03-01 11:40:29 +01:00
|
|
|
|
VERSION=$(docker-compose exec -T "$SERVICE" cat package.json | jq -r '.version')
|
2020-10-06 14:18:35 +02:00
|
|
|
|
fi
|
2021-06-11 13:47:25 +02:00
|
|
|
|
target_version
|
|
|
|
|
if [ "$TARGET" = 'custom' ]; then return; fi
|
2020-10-06 14:18:35 +02:00
|
|
|
|
|
2021-06-11 13:47:25 +02:00
|
|
|
|
if verlt "$VERSION" 2.8.3 && verlt 2.8.3 "$TARGET"; then
|
|
|
|
|
version_error "v2.8.3 first"
|
|
|
|
|
elif verlt "$VERSION" 3.1.2 && verlt 3.1.2 "$TARGET"; then
|
|
|
|
|
version_error "v3.1.2 first"
|
|
|
|
|
elif verlt "$VERSION" 4.0.4 && verlt 4.0.4 "$TARGET"; then
|
|
|
|
|
version_error "v4.0.4 first"
|
|
|
|
|
elif verlt "$VERSION" 4.4.6 && verlt 4.4.6 "$TARGET"; then
|
|
|
|
|
version_error "v4.4.6 first"
|
|
|
|
|
elif verlt "$VERSION" 4.7.13 && verlt 4.7.13 "$TARGET"; then
|
|
|
|
|
version_error "v4.7.13 first"
|
|
|
|
|
elif verlt "$TARGET" "$VERSION"; then
|
|
|
|
|
version_error "a version > $VERSION"
|
2020-10-06 14:18:35 +02:00
|
|
|
|
fi
|
|
|
|
|
}
|
|
|
|
|
|
2020-06-30 16:11:12 +02:00
|
|
|
|
add_environments()
|
|
|
|
|
{
|
|
|
|
|
for ENV in "${ENVIRONMENTS[@]}"; do
|
|
|
|
|
if [[ "$ENV" =~ ^[A-Z0-9_]+=.*$ ]]; then
|
2021-03-24 11:04:59 +01:00
|
|
|
|
printf "\e[91m::\e[0m \e[1mInserting variable %s..\e[0m.\n" "$ENV"
|
2020-06-30 16:11:12 +02:00
|
|
|
|
printf "# added on %s\n%s\n" "$(date +%Y-%m-%d\ %R)" "$ENV" >> "config/env"
|
|
|
|
|
else
|
2021-05-17 15:42:23 +02:00
|
|
|
|
printf "\e[93m[ ⚠ ] Ignoring invalid option: -e %s.\e[39m\n Given value is not valid environment variable, please see http://env.doc.fab.mn\n" "$ENV"
|
2020-06-30 16:11:12 +02:00
|
|
|
|
fi
|
|
|
|
|
done
|
|
|
|
|
}
|
|
|
|
|
|
2021-03-08 17:08:07 +01:00
|
|
|
|
clean_env_file()
|
|
|
|
|
{
|
|
|
|
|
# docker run --env-file does not support whitespaces in the environment variables so we must clean the file
|
|
|
|
|
sed -ri 's/^([A-Z0-9_]+)\s*=\s*(.*)$/\1=\2/g' ./config/env
|
|
|
|
|
}
|
|
|
|
|
|
2020-10-23 12:15:25 +02:00
|
|
|
|
compile_assets()
|
2020-10-21 10:44:43 +02:00
|
|
|
|
{
|
2021-03-01 10:13:48 +01:00
|
|
|
|
IMAGE=$(yq eval '.services.*.image | select(. == "sleede/fab-manager*")' docker-compose.yml)
|
|
|
|
|
mapfile -t COMPOSE_ENVS < <(yq eval ".services.$SERVICE.environment" docker-compose.yml)
|
2020-10-21 13:13:13 +02:00
|
|
|
|
ENV_ARGS=$(for i in "${COMPOSE_ENVS[@]}"; do sed 's/: /=/g;s/^/-e /g' <<< "$i"; done)
|
|
|
|
|
PG_ID=$(docker-compose ps -q postgres)
|
2020-10-23 12:15:25 +02:00
|
|
|
|
if [[ "$PG_ID" = "" ]]; then
|
2021-06-11 13:47:25 +02:00
|
|
|
|
restore_tag
|
2021-04-19 17:07:03 +02:00
|
|
|
|
printf "\e[91m[ ❌ ] PostgreSQL container is not running, unable to compile the assets\e[39m\nExiting..."
|
2021-06-01 09:03:24 +02:00
|
|
|
|
exit 4
|
2020-10-23 12:15:25 +02:00
|
|
|
|
fi
|
2020-10-21 13:13:13 +02:00
|
|
|
|
PG_NET_ID=$(docker inspect "$PG_ID" -f "{{json .NetworkSettings.Networks }}" | jq -r '.[] .NetworkID')
|
2021-03-08 17:08:07 +01:00
|
|
|
|
clean_env_file
|
2020-10-21 13:13:13 +02:00
|
|
|
|
# shellcheck disable=SC2068
|
2021-05-18 16:18:44 +02:00
|
|
|
|
if ! docker run --rm --env-file ./config/env ${ENV_ARGS[@]} --link "$PG_ID" --net "$PG_NET_ID" -v "${PWD}/public/new_packs:/usr/src/app/public/packs" "$IMAGE" bundle exec rake assets:precompile; then
|
2021-06-11 13:47:25 +02:00
|
|
|
|
restore_tag
|
2021-05-18 16:18:44 +02:00
|
|
|
|
printf "\e[91m[ ❌ ] Something went wrong while compiling the assets, please check the logs above.\e[39m\nExiting...\n"
|
|
|
|
|
exit 4
|
|
|
|
|
fi
|
2020-10-21 10:44:43 +02:00
|
|
|
|
docker-compose down
|
|
|
|
|
rm -rf public/packs
|
|
|
|
|
mv public/new_packs public/packs
|
|
|
|
|
}
|
|
|
|
|
|
2021-06-11 13:47:25 +02:00
|
|
|
|
force_version()
|
|
|
|
|
{
|
|
|
|
|
if [ "$FORCE_TARGET" != "true" ]; then return; fi
|
|
|
|
|
|
|
|
|
|
yq -i eval ".services.$SERVICE.image = \"sleede/fab-manager:release-v$TARGET\"" docker-compose.yml
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
restore_tag()
|
|
|
|
|
{
|
|
|
|
|
if [ "$FORCE_TARGET" != "true" ]; then return; fi
|
|
|
|
|
|
|
|
|
|
yq -i eval ".services.$SERVICE.image = \"sleede/fab-manager$TAG\"" docker-compose.yml
|
|
|
|
|
}
|
|
|
|
|
|
2020-06-30 16:11:12 +02:00
|
|
|
|
upgrade()
|
|
|
|
|
{
|
2021-06-11 13:47:25 +02:00
|
|
|
|
[[ "$YES_ALL" = "true" ]] && confirm="y" || read -rp "[91m::[0m [1mProceed with upgrading to version $TARGET ?[0m (Y/n) " confirm </dev/tty
|
2021-04-14 16:18:35 +02:00
|
|
|
|
if [[ "$confirm" = "n" ]]; then exit 2; fi
|
2020-06-30 16:11:12 +02:00
|
|
|
|
|
2021-04-14 16:18:35 +02:00
|
|
|
|
add_environments
|
2021-06-11 13:47:25 +02:00
|
|
|
|
force_version
|
2021-04-19 17:07:03 +02:00
|
|
|
|
if ! docker-compose pull "$SERVICE"; then
|
2021-06-11 13:47:25 +02:00
|
|
|
|
restore_tag
|
2021-04-19 17:07:03 +02:00
|
|
|
|
printf "\e[91m[ ❌ ] An error occurred, detected service name: %s\e[39m\nExiting..." "$SERVICE"
|
2021-06-01 09:03:24 +02:00
|
|
|
|
exit 4
|
2020-06-30 16:11:12 +02:00
|
|
|
|
fi
|
2020-10-21 15:18:39 +02:00
|
|
|
|
BRANCH='master'
|
2021-03-01 10:13:48 +01:00
|
|
|
|
if yq eval '.services.*.image | select(. == "sleede/fab-manager*")' docker-compose.yml | grep -q ':dev'; then BRANCH='dev'; fi
|
2020-06-30 16:11:12 +02:00
|
|
|
|
for SCRIPT in "${SCRIPTS[@]}"; do
|
2021-03-24 11:04:59 +01:00
|
|
|
|
printf "\e[91m::\e[0m \e[1mRunning script %s from branch %s...\e[0m\n" "$SCRIPT" "$BRANCH"
|
2020-07-01 10:52:43 +02:00
|
|
|
|
if [[ "$YES_ALL" = "true" ]]; then
|
2020-10-21 15:18:39 +02:00
|
|
|
|
\curl -sSL "https://raw.githubusercontent.com/sleede/fab-manager/$BRANCH/scripts/$SCRIPT.sh" | bash -s -- -y
|
2020-07-01 10:52:43 +02:00
|
|
|
|
else
|
2020-10-21 15:18:39 +02:00
|
|
|
|
\curl -sSL "https://raw.githubusercontent.com/sleede/fab-manager/$BRANCH/scripts/$SCRIPT.sh" | bash
|
2020-07-01 10:52:43 +02:00
|
|
|
|
fi
|
2021-04-19 17:07:03 +02:00
|
|
|
|
# shellcheck disable=SC2181
|
|
|
|
|
if [[ $? != 0 ]]; then
|
|
|
|
|
printf "\e[93m[ ⚠ ] Something may have went wrong while running \"%s\", please check the logs above...\e[39m\n" "$SCRIPT"
|
|
|
|
|
[[ "$YES_ALL" = "true" ]] && confirm="y" || read -rp "[91m::[0m [1mIgnore and continue?[0m (Y/n) " confirm </dev/tty
|
2021-06-11 13:47:25 +02:00
|
|
|
|
if [[ "$confirm" = "n" ]]; then restore_tag; exit 4; fi
|
2021-04-19 17:07:03 +02:00
|
|
|
|
fi
|
2020-06-30 16:11:12 +02:00
|
|
|
|
done
|
2021-06-01 08:54:19 +02:00
|
|
|
|
for PRE in "${PREPROCESSING[@]}"; do
|
|
|
|
|
printf "\e[91m::\e[0m \e[1mRunning preprocessing command %s...\e[0m\n" "$PRE"
|
2021-06-10 17:24:16 +02:00
|
|
|
|
if ! docker-compose run --rm "$SERVICE" bundle exec "$PRE" </dev/tty; then
|
2021-06-11 13:47:25 +02:00
|
|
|
|
restore_tag
|
2021-06-01 08:54:19 +02:00
|
|
|
|
printf "\e[91m[ ❌ ] Something went wrong while running \"%s\", please check the logs above.\e[39m\nExiting...\n" "$PRE"
|
|
|
|
|
exit 4
|
|
|
|
|
fi
|
|
|
|
|
done
|
2020-10-23 12:15:25 +02:00
|
|
|
|
compile_assets
|
2021-04-19 17:07:03 +02:00
|
|
|
|
if ! docker-compose run --rm "$SERVICE" bundle exec rake db:migrate; then
|
2021-06-11 13:47:25 +02:00
|
|
|
|
restore_tag
|
2021-04-19 17:07:03 +02:00
|
|
|
|
printf "\e[91m[ ❌ ] Something went wrong while migrating the database, please check the logs above.\e[39m\nExiting...\n"
|
|
|
|
|
exit 4
|
|
|
|
|
fi
|
2020-06-30 16:11:12 +02:00
|
|
|
|
for COMMAND in "${COMMANDS[@]}"; do
|
2021-03-24 11:04:59 +01:00
|
|
|
|
printf "\e[91m::\e[0m \e[1mRunning command %s...\e[0m\n" "$COMMAND"
|
2021-06-10 17:24:16 +02:00
|
|
|
|
if ! docker-compose run --rm "$SERVICE" bundle exec "$COMMAND" </dev/tty; then
|
2021-06-11 13:47:25 +02:00
|
|
|
|
restore_tag
|
2021-04-19 17:07:03 +02:00
|
|
|
|
printf "\e[91m[ ❌ ] Something went wrong while running \"%s\", please check the logs above.\e[39m\nExiting...\n" "$COMMAND"
|
|
|
|
|
exit 4
|
|
|
|
|
fi
|
2020-06-30 16:11:12 +02:00
|
|
|
|
done
|
|
|
|
|
docker-compose up -d
|
2021-06-11 13:47:25 +02:00
|
|
|
|
restore_tag
|
2020-06-30 16:11:12 +02:00
|
|
|
|
docker ps
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
clean()
|
|
|
|
|
{
|
2021-03-24 11:04:59 +01:00
|
|
|
|
echo -e "\e[91m::\e[0m \e[1mCurrent disk usage:\e[0m"
|
2020-06-30 16:11:12 +02:00
|
|
|
|
df -h /
|
2021-04-19 17:07:03 +02:00
|
|
|
|
[[ "$YES_ALL" = "true" ]] && confirm="y" || read -rp "[91m::[0m [1mClean previous docker images?[0m (y/N) " confirm </dev/tty
|
2020-06-30 16:11:12 +02:00
|
|
|
|
if [[ "$confirm" == "y" ]]; then
|
|
|
|
|
/usr/bin/docker image prune -f
|
|
|
|
|
fi
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
usage()
|
|
|
|
|
{
|
|
|
|
|
printf "Usage: %s [OPTIONS]
|
|
|
|
|
Options:
|
|
|
|
|
-h Print this message and quit
|
|
|
|
|
-y Answer yes to all questions
|
2021-06-11 13:47:25 +02:00
|
|
|
|
-t <string> Force the upgrade to target the specified version
|
2021-06-01 08:54:19 +02:00
|
|
|
|
-p <string> Run the preprocessing command (TODO DEPLOY)
|
2020-06-30 16:11:12 +02:00
|
|
|
|
-c <string> Provides additional upgrade command, run in the context of the app (TODO DEPLOY)
|
|
|
|
|
-s <string> Executes a remote script (TODO DEPOY)
|
2021-06-01 09:03:24 +02:00
|
|
|
|
-e <string> Adds the environment variable to config/env\n" "$(basename "$0")
|
|
|
|
|
Return codes:
|
|
|
|
|
0 Upgrade terminated successfully
|
|
|
|
|
1 Configuration required
|
|
|
|
|
2 Aborted by user
|
|
|
|
|
3 Version not supported
|
|
|
|
|
4 Unexpected error" 1>&2
|
2020-06-30 16:11:12 +02:00
|
|
|
|
exit 1
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function trap_ctrlc()
|
|
|
|
|
{
|
|
|
|
|
echo "Ctrl^C, exiting..."
|
|
|
|
|
exit 2
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
proceed()
|
|
|
|
|
{
|
|
|
|
|
trap "trap_ctrlc" 2 # SIGINT
|
|
|
|
|
parseparams "$@"
|
|
|
|
|
config
|
2020-10-06 14:18:35 +02:00
|
|
|
|
version_check
|
2020-06-30 16:11:12 +02:00
|
|
|
|
upgrade
|
|
|
|
|
clean
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
proceed "$@"
|