2017-08-23 22:15:42 +02:00
#!/usr/bin/env bash
set -e
# Setup
2022-02-24 16:36:59 +01:00
if command -v docker-compose & > /dev/null
then
dccmd = 'docker-compose'
else
dccmd = 'docker compose'
fi
2017-08-23 22:15:42 +02:00
2018-05-31 18:05:26 +02:00
CYAN = '\033[0;36m'
2022-02-23 21:35:36 +01:00
RED = '\033[1;31m'
2018-05-31 18:05:26 +02:00
NC = '\033[0m' # No Color
2017-08-23 22:15:42 +02:00
DIR = " $( cd " $( dirname " ${ BASH_SOURCE [0] } " ) " && pwd ) "
2018-05-31 18:05:26 +02:00
OUTPUT_DIR = ".."
2017-08-23 22:51:36 +02:00
if [ $# -gt 1 ]
2017-08-23 22:15:42 +02:00
then
OUTPUT_DIR = $2
fi
2017-11-09 04:24:23 +01:00
COREVERSION = "latest"
2017-08-24 02:05:53 +02:00
if [ $# -gt 2 ]
2017-08-23 22:15:42 +02:00
then
2017-11-09 04:24:23 +01:00
COREVERSION = $3
2017-08-23 22:15:42 +02:00
fi
2017-11-09 04:24:23 +01:00
WEBVERSION = "latest"
if [ $# -gt 3 ]
then
WEBVERSION = $4
fi
2021-12-29 17:07:16 +01:00
KEYCONNECTORVERSION = "latest"
if [ $# -gt 4 ]
then
KEYCONNECTORVERSION = $5
fi
2017-11-09 04:24:23 +01:00
OS = "lin"
2018-05-31 18:05:26 +02:00
[ " $( uname) " = = "Darwin" ] && OS = "mac"
ENV_DIR = " $OUTPUT_DIR /env "
DOCKER_DIR = " $OUTPUT_DIR /docker "
2018-05-31 19:49:41 +02:00
# Initialize UID/GID which will be used to run services from within containers
2018-05-31 18:05:26 +02:00
if ! grep -q "^LOCAL_UID=" $ENV_DIR /uid.env 2>/dev/null || ! grep -q "^LOCAL_GID=" $ENV_DIR /uid.env 2>/dev/null
2017-08-23 22:15:42 +02:00
then
2018-05-31 18:05:26 +02:00
LUID = " LOCAL_UID=`id -u $USER ` "
[ " $LUID " = = "LOCAL_UID=0" ] && LUID = "LOCAL_UID=65534"
LGID = " LOCAL_GID=`id -g $USER ` "
[ " $LGID " = = "LOCAL_GID=0" ] && LGID = "LOCAL_GID=65534"
mkdir -p $ENV_DIR
echo $LUID >$ENV_DIR /uid.env
echo $LGID >>$ENV_DIR /uid.env
2017-08-23 22:15:42 +02:00
fi
# Functions
2018-05-31 18:05:26 +02:00
function install( ) {
LETS_ENCRYPT = "n"
2019-05-23 02:07:39 +02:00
echo -e -n " ${ CYAN } (!) ${ NC } Enter the domain name for your Bitwarden instance (ex. bitwarden.example.com): "
2018-05-31 18:05:26 +02:00
read DOMAIN
echo ""
if [ " $DOMAIN " = = "" ]
then
DOMAIN = "localhost"
fi
if [ " $DOMAIN " != "localhost" ]
then
echo -e -n " ${ CYAN } (!) ${ NC } Do you want to use Let's Encrypt to generate a free SSL certificate? (y/n): "
read LETS_ENCRYPT
echo ""
if [ " $LETS_ENCRYPT " = = "y" ]
then
echo -e -n " ${ CYAN } (!) ${ NC } Enter your email address (Let's Encrypt will send you certificate expiration reminders): "
read EMAIL
echo ""
mkdir -p $OUTPUT_DIR /letsencrypt
docker pull certbot/certbot
docker run -it --rm --name certbot -p 80:80 -v $OUTPUT_DIR /letsencrypt:/etc/letsencrypt/ certbot/certbot \
2018-05-31 19:49:41 +02:00
certonly --standalone --noninteractive --agree-tos --preferred-challenges http \
--email $EMAIL -d $DOMAIN --logs-dir /etc/letsencrypt/logs
2018-05-31 18:05:26 +02:00
fi
fi
2021-07-02 16:52:34 +02:00
echo -e -n " ${ CYAN } (!) ${ NC } Enter the database name for your Bitwarden instance (ex. vault): "
read DATABASE
echo ""
if [ " $DATABASE " = = "" ]
then
DATABASE = "vault"
fi
2018-05-31 18:05:26 +02:00
pullSetup
2018-05-31 19:49:41 +02:00
docker run -it --rm --name setup -v $OUTPUT_DIR :/bitwarden \
--env-file $ENV_DIR /uid.env bitwarden/setup:$COREVERSION \
dotnet Setup.dll -install 1 -domain $DOMAIN -letsencrypt $LETS_ENCRYPT -os $OS \
2021-12-29 17:07:16 +01:00
-corev $COREVERSION -webv $WEBVERSION -dbname " $DATABASE " -keyconnectorv $KEYCONNECTORVERSION
2018-05-31 18:05:26 +02:00
}
2017-08-23 22:15:42 +02:00
function dockerComposeUp( ) {
2018-08-31 15:16:14 +02:00
dockerComposeFiles
2020-05-30 00:03:55 +02:00
dockerComposeVolumes
2022-02-24 16:36:59 +01:00
$dccmd up -d
2017-08-23 22:15:42 +02:00
}
function dockerComposeDown( ) {
2018-08-31 15:16:14 +02:00
dockerComposeFiles
2022-02-24 16:36:59 +01:00
if [ $( $dccmd ps | wc -l) -gt 2 ] ; then
$dccmd down
2021-01-21 20:49:27 +01:00
fi
2017-08-23 22:15:42 +02:00
}
2017-08-27 04:36:25 +02:00
function dockerComposePull( ) {
2018-08-31 15:16:14 +02:00
dockerComposeFiles
2022-02-24 16:36:59 +01:00
$dccmd pull
2018-08-31 15:16:14 +02:00
}
function dockerComposeFiles( ) {
2018-02-20 17:15:16 +01:00
if [ -f " ${ DOCKER_DIR } /docker-compose.override.yml " ]
2018-02-20 17:09:47 +01:00
then
2018-08-31 15:16:14 +02:00
export COMPOSE_FILE = " $DOCKER_DIR /docker-compose.yml: $DOCKER_DIR /docker-compose.override.yml "
2018-02-20 17:09:47 +01:00
else
2018-08-31 15:16:14 +02:00
export COMPOSE_FILE = " $DOCKER_DIR /docker-compose.yml "
2018-02-20 17:09:47 +01:00
fi
2018-11-09 03:32:23 +01:00
export COMPOSE_HTTP_TIMEOUT = "300"
2017-08-27 04:36:25 +02:00
}
2020-05-30 00:03:55 +02:00
function dockerComposeVolumes( ) {
createDir "core"
createDir "core/attachments"
createDir "logs"
createDir "logs/admin"
createDir "logs/api"
createDir "logs/events"
createDir "logs/icons"
createDir "logs/identity"
createDir "logs/mssql"
createDir "logs/nginx"
createDir "logs/notifications"
2020-09-14 19:08:43 +02:00
createDir "logs/sso"
2021-10-15 10:40:43 +02:00
createDir "logs/portal"
2020-05-30 00:03:55 +02:00
createDir "mssql/backups"
createDir "mssql/data"
}
function createDir( ) {
if [ ! -d " ${ OUTPUT_DIR } / $1 " ]
then
echo " Creating directory $OUTPUT_DIR / $1 "
mkdir -p $OUTPUT_DIR /$1
fi
}
2017-08-23 22:15:42 +02:00
function dockerPrune( ) {
2018-11-07 15:51:16 +01:00
docker image prune --all --force --filter= "label=com.bitwarden.product=bitwarden" \
--filter= "label!=com.bitwarden.project=setup"
2017-08-23 22:15:42 +02:00
}
function updateLetsEncrypt( ) {
2017-11-09 04:24:23 +01:00
if [ -d " ${ OUTPUT_DIR } /letsencrypt/live " ]
2017-08-23 22:15:42 +02:00
then
2017-08-27 04:54:10 +02:00
docker pull certbot/certbot
2018-03-27 20:55:33 +02:00
docker run -i --rm --name certbot -p 443:443 -p 80:80 \
-v $OUTPUT_DIR /letsencrypt:/etc/letsencrypt/ certbot/certbot \
2017-08-23 22:15:42 +02:00
renew --logs-dir /etc/letsencrypt/logs
fi
}
2020-06-17 15:12:01 +02:00
function forceUpdateLetsEncrypt( ) {
if [ -d " ${ OUTPUT_DIR } /letsencrypt/live " ]
then
docker pull certbot/certbot
docker run -i --rm --name certbot -p 443:443 -p 80:80 \
-v $OUTPUT_DIR /letsencrypt:/etc/letsencrypt/ certbot/certbot \
renew --logs-dir /etc/letsencrypt/logs --force-renew
fi
2020-06-17 15:05:35 +02:00
}
2017-08-23 22:15:42 +02:00
function updateDatabase( ) {
2017-10-25 23:21:35 +02:00
pullSetup
2019-10-02 16:20:33 +02:00
dockerComposeFiles
2022-02-24 16:36:59 +01:00
MSSQL_ID = $( $dccmd ps -q mssql)
2019-10-02 16:20:33 +02:00
docker run -i --rm --name setup --network container:$MSSQL_ID \
2018-05-31 18:05:26 +02:00
-v $OUTPUT_DIR :/bitwarden --env-file $ENV_DIR /uid.env bitwarden/setup:$COREVERSION \
2021-12-29 17:07:16 +01:00
dotnet Setup.dll -update 1 -db 1 -os $OS -corev $COREVERSION -webv $WEBVERSION -keyconnectorv $KEYCONNECTORVERSION
2017-08-23 22:15:42 +02:00
echo "Database update complete"
}
2021-01-21 20:49:40 +01:00
function updatebw( ) {
2021-12-30 19:27:25 +01:00
KEY_CONNECTOR_ENABLED = $( grep -A3 'enable_key_connector:' $OUTPUT_DIR /config.yml | tail -n1 | awk '{ print $2}' )
2022-02-24 16:36:59 +01:00
CORE_ID = $( $dccmd ps -q admin)
WEB_ID = $( $dccmd ps -q web)
2022-02-09 21:12:44 +01:00
if [ " $KEY_CONNECTOR_ENABLED " = true ] ;
2021-12-30 19:27:25 +01:00
then
2022-02-24 16:36:59 +01:00
KEYCONNECTOR_ID = $( $dccmd ps -q key-connector)
2021-12-30 19:27:25 +01:00
fi
if [ $KEYCONNECTOR_ID ] &&
docker inspect --format= '{{.Config.Image}}:' $CORE_ID | grep -F " : $COREVERSION : " | grep -q " :[0-9.]*: $" &&
docker inspect --format= '{{.Config.Image}}:' $WEB_ID | grep -F " : $WEBVERSION : " | grep -q " :[0-9.]*: $" &&
docker inspect --format= '{{.Config.Image}}:' $KEYCONNECTOR_ID | grep -F " : $KEYCONNECTORVERSION : " | grep -q " :[0-9.]*: $"
then
echo "Update not needed"
exit
elif
docker inspect --format= '{{.Config.Image}}:' $CORE_ID | grep -F " : $COREVERSION : " | grep -q " :[0-9.]*: $" &&
docker inspect --format= '{{.Config.Image}}:' $WEB_ID | grep -F " : $WEBVERSION : " | grep -q " :[0-9.]*: $"
2021-01-21 20:49:40 +01:00
then
echo "Update not needed"
exit
fi
dockerComposeDown
update withpull
restart
dockerPrune
echo "Pausing 60 seconds for database to come online. Please wait..."
sleep 60
}
2017-10-25 23:21:35 +02:00
function update( ) {
2018-08-30 22:25:33 +02:00
if [ " $1 " = = "withpull" ]
then
pullSetup
fi
2018-05-31 18:05:26 +02:00
docker run -i --rm --name setup -v $OUTPUT_DIR :/bitwarden \
--env-file $ENV_DIR /uid.env bitwarden/setup:$COREVERSION \
2021-12-29 17:07:16 +01:00
dotnet Setup.dll -update 1 -os $OS -corev $COREVERSION -webv $WEBVERSION -keyconnectorv $KEYCONNECTORVERSION
2017-10-25 23:21:35 +02:00
}
2022-02-23 21:35:36 +01:00
function uninstall( ) {
echo -e -n " ${ RED } (WARNING: UNINSTALL STARTED) Would you like to save the database files? (y/n): ${ NC } "
read KEEP_DATABASE
if [ " $KEEP_DATABASE " = = "y" ]
then
echo "Saving database files."
tar -cvzf "./bitwarden_database.tar.gz" " $OUTPUT_DIR /mssql "
echo -e -n " ${ RED } (SAVED DATABASE FILES: YES): WARNING: ALL DATA WILL BE REMOVED, INCLUDING THE FOLDER $OUTPUT_DIR ): Are you sure you want to uninstall Bitwarden? (y/n): ${ NC } "
read UNINSTALL_ACTION
else
echo -e -n " ${ RED } WARNING: ALL DATA WILL BE REMOVED, INCLUDING THE FOLDER $OUTPUT_DIR ): Are you sure you want to uninstall Bitwarden? (y/n): ${ NC } "
read UNINSTALL_ACTION
fi
if [ " $UNINSTALL_ACTION " = = "y" ]
then
echo "Uninstalling Bitwarden..."
dockerComposeDown
echo " Removing $OUTPUT_DIR "
rm -R $OUTPUT_DIR
echo "Removing MSSQL docker volume."
docker volume prune --force --filter= "label=com.bitwarden.product=bitwarden"
echo "Bitwarden uninstall complete!"
else
echo -e -n " ${ CYAN } (!) Bitwarden uninstall canceled. ${ NC } "
exit 1
fi
echo -e -n " ${ RED } (!) Would you like to purge all local Bitwarden container images? (y/n): ${ NC } "
read PURGE_ACTION
if [ " $PURGE_ACTION " = = "y" ]
then
dockerPrune
echo -e -n " ${ CYAN } Bitwarden uninstall complete! ${ NC } "
fi
}
2017-08-24 17:16:01 +02:00
function printEnvironment( ) {
2017-10-25 23:21:35 +02:00
pullSetup
2018-05-31 18:05:26 +02:00
docker run -i --rm --name setup -v $OUTPUT_DIR :/bitwarden \
--env-file $ENV_DIR /uid.env bitwarden/setup:$COREVERSION \
2021-12-29 17:07:16 +01:00
dotnet Setup.dll -printenv 1 -os $OS -corev $COREVERSION -webv $WEBVERSION -keyconnectorv $KEYCONNECTORVERSION
2017-08-24 17:16:01 +02:00
}
2017-10-25 23:21:35 +02:00
function restart( ) {
2017-08-23 22:15:42 +02:00
dockerComposeDown
2017-08-27 04:36:25 +02:00
dockerComposePull
2017-08-23 22:15:42 +02:00
updateLetsEncrypt
dockerComposeUp
2017-08-24 17:16:01 +02:00
printEnvironment
2017-10-25 23:21:35 +02:00
}
2020-06-17 15:12:01 +02:00
function certRestart( ) {
2020-06-17 15:05:35 +02:00
dockerComposeDown
dockerComposePull
2020-06-17 15:12:01 +02:00
forceUpdateLetsEncrypt
2020-06-17 15:05:35 +02:00
dockerComposeUp
printEnvironment
}
2017-10-25 23:21:35 +02:00
function pullSetup( ) {
2017-11-09 04:24:23 +01:00
docker pull bitwarden/setup:$COREVERSION
2017-10-25 23:21:35 +02:00
}
# Commands
2021-01-21 20:49:40 +01:00
case $1 in
"install" )
install
; ;
"start" | "restart" )
restart
; ;
"pull" )
dockerComposePull
; ;
"stop" )
dockerComposeDown
; ;
"renewcert" )
certRestart
; ;
"updateconf" )
dockerComposeDown
update withpull
; ;
"updatedb" )
updateDatabase
; ;
"update" )
dockerComposeFiles
updatebw
updateDatabase
; ;
2022-02-23 21:35:36 +01:00
"uninstall" )
dockerComposeFiles
uninstall
; ;
2021-01-21 20:49:40 +01:00
"rebuild" )
dockerComposeDown
update nopull
; ;
esac