2023-12-21 15:30:31 +00:00
#!/bin/bash
2018-11-30 15:24:19 +00:00
if [ -z " ${ NONINTERACTIVE :- } " ] ; then
2014-08-25 12:09:37 +00:00
# Install 'dialog' so we can ask the user questions. The original motivation for
# this was being able to ask the user for input even if stdin has been redirected,
# e.g. if we piped a bootstrapping install script to bash to get started. In that
# case, the nifty '[ -t 0 ]' test won't work. But with Vagrant we must suppress so we
2016-01-03 15:48:23 +00:00
# use a shell flag instead. Really suppress any output from installing dialog.
2015-05-03 14:44:37 +00:00
#
2016-01-03 15:48:23 +00:00
# Also install dependencies needed to validate the email address.
2015-06-10 13:43:22 +00:00
if [ ! -f /usr/bin/dialog ] || [ ! -f /usr/bin/python3 ] || [ ! -f /usr/bin/pip3 ] ; then
2023-12-21 16:44:22 +00:00
echo "Installing packages needed for setup..."
2015-06-10 13:43:22 +00:00
apt-get -q -q update
apt_get_quiet install dialog python3 python3-pip || exit 1
fi
2015-05-03 14:44:37 +00:00
2018-01-20 15:54:56 +00:00
# Installing email_validator is repeated in setup/management.sh, but in setup/management.sh
# we install it inside a virtualenv. In this script, we don't have the virtualenv yet
# so we install the python package globally.
2015-11-18 14:43:08 +00:00
hide_output pip3 install "email_validator>=1.0.0" || exit 1
2015-05-03 14:44:37 +00:00
2014-08-25 12:09:37 +00:00
message_box "Mail-in-a-Box Installation" \
" Hello and thanks for deploying a Mail-in-a-Box!
\n \n I' m going to ask you a few questions.
2016-02-14 18:40:43 +00:00
\n \n To change your answers later, just run 'sudo mailinabox' from the command line.
2016-02-14 19:24:00 +00:00
\n \n NOTE: You should only install this on a brand new Ubuntu installation 100% dedicated to Mail-in-a-Box. Mail-in-a-Box will, for example, remove apache2."
2014-08-25 12:09:37 +00:00
fi
# The box needs a name.
2024-12-24 14:36:34 +00:00
if [ -z " ${ BOX_HOSTNAME :- } " ] ; then
if [ -z " ${ DEFAULT_BOX_HOSTNAME :- } " ] ; then
2014-08-25 12:26:39 +00:00
# We recommend to use box.example.com as this hosts name. The
# domain the user possibly wants to use is example.com then.
# We strip the string "box." from the hostname to get the mail
# domain. If the hostname differs, nothing happens here.
2023-12-21 16:08:32 +00:00
DEFAULT_DOMAIN_GUESS = $( get_default_hostname | sed -e 's/^box\.//' )
2014-08-25 12:26:39 +00:00
2014-08-25 12:09:37 +00:00
# This is the first run. Ask the user for his email address so we can
# provide the best default for the box's hostname.
input_box "Your Email Address" \
" What email address are you setting this box up to manage?
\n \n The part after the @-sign must be a domain name or subdomain
that you control. You can add other email addresses to this
box later ( including email addresses on other domain names
or subdomains you control) .
\n \n We' ve guessed an email address. Backspace it and type in what
you really want.
\n \n Email Address:" \
2014-08-25 12:26:39 +00:00
" me@ $DEFAULT_DOMAIN_GUESS " \
2014-08-25 12:09:37 +00:00
EMAIL_ADDR
if [ -z " $EMAIL_ADDR " ] ; then
# user hit ESC/cancel
exit
fi
2018-01-20 15:54:56 +00:00
while ! python3 management/mailconfig.py validate-email " $EMAIL_ADDR "
2014-08-25 12:09:37 +00:00
do
input_box "Your Email Address" \
"That's not a valid email address.\n\nWhat email address are you setting this box up to manage?" \
2023-12-21 14:58:34 +00:00
" $EMAIL_ADDR " \
2014-08-25 12:09:37 +00:00
EMAIL_ADDR
if [ -z " $EMAIL_ADDR " ] ; then
# user hit ESC/cancel
exit
fi
done
# Take the part after the @-sign as the user's domain name, and add
# 'box.' to the beginning to create a default hostname for this machine.
2024-12-24 14:36:34 +00:00
DEFAULT_BOX_HOSTNAME = box.$( echo " $EMAIL_ADDR " | sed 's/.*@//' )
2014-08-25 12:09:37 +00:00
fi
input_box "Hostname" \
" This box needs a name, called a 'hostname'. The name will form a part of the box's web address.
\n \n We recommend that the name be a subdomain of the domain in your email
2024-12-24 14:36:34 +00:00
address, so we' re suggesting $DEFAULT_BOX_HOSTNAME .
2014-08-25 12:09:37 +00:00
\n \n You can change it, but we recommend you don' t.
\n \n Hostname:" \
2024-12-24 14:36:34 +00:00
" $DEFAULT_BOX_HOSTNAME " \
BOX_HOSTNAME
2014-08-25 12:09:37 +00:00
2024-12-24 14:36:34 +00:00
if [ -z " $BOX_HOSTNAME " ] ; then
2014-08-25 12:09:37 +00:00
# user hit ESC/cancel
exit
fi
fi
# If the machine is behind a NAT, inside a VM, etc., it may not know
# its IP address on the public network / the Internet. Ask the Internet
# and possibly confirm with user.
2018-11-30 15:24:19 +00:00
if [ -z " ${ PUBLIC_IP :- } " ] ; then
2014-08-25 12:09:37 +00:00
# Ask the Internet.
GUESSED_IP = $( get_publicip_from_web_service 4)
# On the first run, if we got an answer from the Internet then don't
# ask the user.
2023-12-21 16:06:26 +00:00
if [ [ -z " ${ DEFAULT_PUBLIC_IP :- } " && -n " $GUESSED_IP " ] ] ; then
2014-08-25 12:09:37 +00:00
PUBLIC_IP = $GUESSED_IP
# Otherwise on the first run at least provide a default.
2018-12-26 20:39:47 +00:00
elif [ [ -z " ${ DEFAULT_PUBLIC_IP :- } " ] ] ; then
2014-08-25 12:09:37 +00:00
DEFAULT_PUBLIC_IP = $( get_default_privateip 4)
# On later runs, if the previous value matches the guessed value then
# don't ask the user either.
2018-12-26 20:39:47 +00:00
elif [ " ${ DEFAULT_PUBLIC_IP :- } " = = " $GUESSED_IP " ] ; then
2014-08-25 12:09:37 +00:00
PUBLIC_IP = $GUESSED_IP
fi
2018-11-30 15:24:19 +00:00
if [ -z " ${ PUBLIC_IP :- } " ] ; then
2014-08-25 12:09:37 +00:00
input_box "Public IP Address" \
" Enter the public IP address of this machine, as given to you by your ISP.
\n \n Public IP address:" \
2023-12-21 14:58:34 +00:00
" ${ DEFAULT_PUBLIC_IP :- } " \
2014-08-25 12:09:37 +00:00
PUBLIC_IP
if [ -z " $PUBLIC_IP " ] ; then
# user hit ESC/cancel
exit
fi
fi
fi
# Same for IPv6. But it's optional. Also, if it looks like the system
# doesn't have an IPv6, don't ask for one.
2018-11-30 15:24:19 +00:00
if [ -z " ${ PUBLIC_IPV6 :- } " ] ; then
2014-08-25 12:09:37 +00:00
# Ask the Internet.
GUESSED_IP = $( get_publicip_from_web_service 6)
MATCHED = 0
2023-12-21 16:06:26 +00:00
if [ [ -z " ${ DEFAULT_PUBLIC_IPV6 :- } " && -n " $GUESSED_IP " ] ] ; then
2014-08-25 12:09:37 +00:00
PUBLIC_IPV6 = $GUESSED_IP
2018-11-30 15:24:19 +00:00
elif [ [ " ${ DEFAULT_PUBLIC_IPV6 :- } " = = " $GUESSED_IP " ] ] ; then
2014-08-25 12:09:37 +00:00
# No IPv6 entered and machine seems to have none, or what
# the user entered matches what the Internet tells us.
PUBLIC_IPV6 = $GUESSED_IP
MATCHED = 1
2018-11-30 15:24:19 +00:00
elif [ [ -z " ${ DEFAULT_PUBLIC_IPV6 :- } " ] ] ; then
2014-08-25 12:09:37 +00:00
DEFAULT_PUBLIC_IP = $( get_default_privateip 6)
fi
2018-11-30 15:24:19 +00:00
if [ [ -z " ${ PUBLIC_IPV6 :- } " && $MATCHED = = 0 ] ] ; then
2014-08-25 12:09:37 +00:00
input_box "IPv6 Address (Optional)" \
" Enter the public IPv6 address of this machine, as given to you by your ISP.
\n \n Leave blank if the machine does not have an IPv6 address.
2014-08-26 10:34:22 +00:00
\n \n Public IPv6 address:" \
2023-12-21 14:58:34 +00:00
" ${ DEFAULT_PUBLIC_IPV6 :- } " \
2014-08-25 12:09:37 +00:00
PUBLIC_IPV6
2023-12-21 16:49:20 +00:00
if [ ! -n " $PUBLIC_IPV6_EXITCODE " ] ; then
2014-08-25 12:09:37 +00:00
# user hit ESC/cancel
exit
fi
fi
fi
# Get the IP addresses of the local network interface(s) that are connected
# to the Internet. We need these when we want to have services bind only to
# the public network interfaces (not loopback, not tunnel interfaces).
2018-11-30 15:24:19 +00:00
if [ -z " ${ PRIVATE_IP :- } " ] ; then
2014-08-25 12:09:37 +00:00
PRIVATE_IP = $( get_default_privateip 4)
fi
2018-11-30 15:24:19 +00:00
if [ -z " ${ PRIVATE_IPV6 :- } " ] ; then
2014-08-25 12:09:37 +00:00
PRIVATE_IPV6 = $( get_default_privateip 6)
fi
if [ [ -z " $PRIVATE_IP " && -z " $PRIVATE_IPV6 " ] ] ; then
echo
2024-07-21 11:01:25 +00:00
echo "I could not determine the IP or IPv6 address of the network interface"
2014-08-25 12:09:37 +00:00
echo "for connecting to the Internet. Setup must stop."
echo
hostname -I
route
echo
exit
fi
2015-05-07 11:11:21 +00:00
# Automatic configuration, e.g. as used in our Vagrant configuration.
if [ " $PUBLIC_IP " = "auto" ] ; then
# Use a public API to get our public IP address, or fall back to local network configuration.
PUBLIC_IP = $( get_publicip_from_web_service 4 || get_default_privateip 4)
fi
if [ " $PUBLIC_IPV6 " = "auto" ] ; then
# Use a public API to get our public IPv6 address, or fall back to local network configuration.
PUBLIC_IPV6 = $( get_publicip_from_web_service 6 || get_default_privateip 6)
fi
2024-12-24 14:36:34 +00:00
if [ " $BOX_HOSTNAME " = "auto" ] ; then
BOX_HOSTNAME = $( get_default_hostname)
2015-05-07 11:11:21 +00:00
fi
# Set STORAGE_USER and STORAGE_ROOT to default values (user-data and /home/user-data), unless
# we've already got those values from a previous run.
2018-11-30 15:24:19 +00:00
if [ -z " ${ STORAGE_USER :- } " ] ; then
STORAGE_USER = $( [ [ -z " ${ DEFAULT_STORAGE_USER :- } " ] ] && echo "user-data" || echo " $DEFAULT_STORAGE_USER " )
2015-05-07 11:11:21 +00:00
fi
2018-11-30 15:24:19 +00:00
if [ -z " ${ STORAGE_ROOT :- } " ] ; then
STORAGE_ROOT = $( [ [ -z " ${ DEFAULT_STORAGE_ROOT :- } " ] ] && echo " /home/ $STORAGE_USER " || echo " $DEFAULT_STORAGE_ROOT " )
2015-05-07 11:11:21 +00:00
fi
# Show the configuration, since the user may have not entered it manually.
echo
2024-12-24 14:36:34 +00:00
echo " Box Hostname: $BOX_HOSTNAME "
2015-05-07 11:11:21 +00:00
echo " Public IP Address: $PUBLIC_IP "
2023-12-21 16:06:26 +00:00
if [ -n " $PUBLIC_IPV6 " ] ; then
2015-05-07 11:11:21 +00:00
echo " Public IPv6 Address: $PUBLIC_IPV6 "
fi
if [ " $PRIVATE_IP " != " $PUBLIC_IP " ] ; then
echo " Private IP Address: $PRIVATE_IP "
fi
if [ " $PRIVATE_IPV6 " != " $PUBLIC_IPV6 " ] ; then
echo " Private IPv6 Address: $PRIVATE_IPV6 "
fi
2015-06-18 11:35:13 +00:00
if [ -f /usr/bin/git ] && [ -d .git ] ; then
2023-12-21 14:58:34 +00:00
echo " Mail-in-a-Box Version: $( git describe --always) "
2015-05-07 11:11:21 +00:00
fi
echo