2016-10-24 16:18:02 +02:00
#!/bin/bash
2016-10-25 12:00:54 -07:00
set -eu -o pipefail
2016-10-24 16:18:02 +02:00
2016-11-01 15:13:20 +01:00
if [[ ${EUID} -ne 0 ]]; then
echo "This script should be run as root." > /dev/stderr
exit 1
fi
2016-10-24 14:52:30 -07:00
# change this to a hash when we make a upgrade release
2016-10-25 14:01:35 +02:00
readonly LOG_FILE="/var/log/cloudron-setup.log"
2017-01-04 10:13:45 -08:00
readonly MINIMUM_DISK_SIZE_GB="20" # this is the size of "/" and required to fit in docker images
2017-01-16 10:36:16 +01:00
readonly MINIMUM_MEMORY="990" # this is mostly reported for 1GB main memory (DO 992, EC2 990)
2017-01-02 18:02:40 +01:00
# copied from cloudron-resize-fs.sh
readonly physical_memory=$(free -m | awk '/Mem:/ { print $2 }')
2017-01-03 09:26:08 +01:00
readonly disk_device="$(for d in $(find /dev -type b); do [ "$(mountpoint -d /)" = "$(mountpoint -x $d)" ] && echo $d && break; done)"
2017-01-19 13:53:50 +01:00
readonly disk_size_bytes=$(fdisk -l ${disk_device} | grep "Disk ${disk_device}" | awk '{ printf "%d", $5 }')
readonly disk_size_gb=$((${disk_size_bytes}/1024/1024/1024))
2017-01-02 18:02:40 +01:00
# verify the system has minimum requirements met
if [[ "${physical_memory}" -lt "${MINIMUM_MEMORY}" ]]; then
2017-01-03 14:27:01 -08:00
echo "Error: Cloudron requires atleast 1GB physical memory"
2017-01-02 18:02:40 +01:00
exit 1
fi
if [[ "${disk_size_gb}" -lt "${MINIMUM_DISK_SIZE_GB}" ]]; then
2017-01-03 14:27:01 -08:00
echo "Error: Cloudron requires atleast 20GB disk space (Disk space on ${disk_device} is ${disk_size_gb}GB)"
2017-01-02 18:02:40 +01:00
exit 1
fi
2016-10-24 14:52:30 -07:00
2016-12-26 11:37:50 -08:00
initBaseImage="true"
# provisioning data
2016-10-24 16:18:02 +02:00
domain=""
2016-10-26 10:53:25 -07:00
provider=""
2016-11-13 11:20:50 +01:00
encryptionKey=""
2016-10-24 14:52:30 -07:00
restoreUrl=""
2016-12-15 07:35:54 -08:00
dnsProvider="manual"
2016-12-05 17:01:23 +01:00
tlsProvider="le-prod"
2016-10-24 14:52:30 -07:00
versionsUrl="https://s3.amazonaws.com/prod-cloudron-releases/versions.json"
2016-12-30 13:12:22 -08:00
requestedVersion="latest"
2017-01-11 12:35:31 -08:00
apiServerOrigin="https://api.cloudron.io"
2016-12-26 11:37:50 -08:00
dataJson=""
2017-01-10 20:19:39 -08:00
prerelease=false
2016-10-24 16:18:02 +02:00
2017-01-17 21:23:57 -08:00
args=$(getopt -o "" -l "domain:,help,skip-baseimage-init,data:,provider:,encryption-key:,restore-url:,tls-provider:,version:,versions-url:,api-server:,dns-provider:,env:,prerelease" -n "$0" -- "$@")
2016-10-24 16:18:02 +02:00
eval set -- "${args}"
while true; do
case "$1" in
--domain) domain="$2"; shift 2;;
2016-10-25 13:34:12 -07:00
--help) echo "See https://cloudron.io/references/selfhosting.html on how to install Cloudron"; exit 0;;
2016-10-24 16:18:02 +02:00
--provider) provider="$2"; shift 2;;
2016-11-13 11:20:50 +01:00
--encryption-key) encryptionKey="$2"; shift 2;;
2016-10-24 14:52:30 -07:00
--restore-url) restoreUrl="$2"; shift 2;;
2016-10-24 17:30:47 +02:00
--tls-provider) tlsProvider="$2"; shift 2;;
2016-12-15 07:35:54 -08:00
--dns-provider) dnsProvider="$2"; shift 2;;
2016-12-30 13:12:22 -08:00
--version) requestedVersion="$2"; shift 2;;
2017-01-10 20:19:39 -08:00
--env)
if [[ "$2" == "dev" ]]; then
2017-01-11 12:35:31 -08:00
apiServerOrigin="https://api.dev.cloudron.io"
2017-01-10 20:19:39 -08:00
versionsUrl="https://s3.amazonaws.com/dev-cloudron-releases/versions.json"
2017-01-12 18:05:42 -08:00
tlsProvider="le-staging"
2017-01-10 20:19:39 -08:00
prerelease="true"
elif [[ "$2" == "staging" ]]; then
2017-01-11 12:35:31 -08:00
apiServerOrigin="https://api.staging.cloudron.io"
2017-01-10 20:19:39 -08:00
versionsUrl="https://s3.amazonaws.com/staging-cloudron-releases/versions.json"
2017-01-12 18:05:42 -08:00
tlsProvider="le-staging"
2017-01-10 20:19:39 -08:00
prerelease="true"
fi
shift 2;;
2016-10-24 14:52:30 -07:00
--versions-url) versionsUrl="$2"; shift 2;;
2017-01-11 12:35:31 -08:00
--api-server) apiServerOrigin="$2"; shift 2;;
2016-12-26 10:00:00 -08:00
--skip-baseimage-init) initBaseImage="false"; shift;;
2016-12-26 11:37:50 -08:00
--data) dataJson="$2"; shift 2;;
2017-01-10 20:19:39 -08:00
--prerelease) prerelease="true"; shift;;
2016-10-24 16:18:02 +02:00
--) break;;
*) echo "Unknown option $1"; exit 1;;
esac
done
2016-12-27 13:36:02 -08:00
# validate arguments in the absence of data
if [[ -z "${dataJson}" ]]; then
if [[ -z "${provider}" ]]; then
echo "--provider is required (generic, scaleway, ec2, digitalocean)"
exit 1
elif [[ \
"${provider}" != "generic" && \
"${provider}" != "scaleway" && \
"${provider}" != "ec2" && \
"${provider}" != "digitalocean" \
]]; then
echo "--provider must be one of: generic, scaleway, ec2, digitalocean"
exit 1
fi
2016-10-26 10:53:25 -07:00
2017-01-12 10:31:54 -08:00
if [[ "${tlsProvider}" != "fallback" && "${tlsProvider}" != "le-prod" && "${tlsProvider}" != "le-staging" ]]; then
echo "--tls-provider must be one of: le-prod, le-staging, fallback"
exit 1
fi
2016-12-27 13:36:02 -08:00
if [[ -z "${dnsProvider}" ]]; then
echo "--dns-provider is required (noop, manual)"
exit 1
elif [[ "${dnsProvider}" != "noop" && "${dnsProvider}" != "manual" ]]; then
echo "--dns-provider must be one of : manual, noop"
exit 1
fi
2016-12-15 07:35:54 -08:00
fi
2016-10-25 15:28:26 +02:00
echo ""
2016-10-25 14:07:28 +02:00
echo "##############################################"
2016-12-30 13:12:22 -08:00
echo " Cloudron Setup (${requestedVersion}) "
2016-10-25 14:07:28 +02:00
echo "##############################################"
echo ""
echo " Follow setup logs in a second terminal with:"
echo " $ tail -f ${LOG_FILE}"
echo ""
2016-12-22 13:28:04 -08:00
echo " Join us at https://chat.cloudron.io for any questions."
echo ""
2016-10-25 14:07:28 +02:00
2017-01-05 14:03:30 -08:00
if [[ "${initBaseImage}" == "true" ]]; then
2017-01-05 14:03:54 -08:00
echo "=> Updating apt and installing script dependancies"
2017-01-05 14:03:30 -08:00
if ! apt-get update &>> "${LOG_FILE}"; then
echo "Could not update package repositories"
exit 1
fi
2017-01-05 14:03:54 -08:00
if ! apt-get install curl python3 ubuntu-standard -y &>> "${LOG_FILE}"; then
2017-01-05 14:03:30 -08:00
echo "Could not install setup dependencies (curl)"
exit 1
fi
fi
2016-10-25 13:05:24 -07:00
echo "=> Checking version"
2017-01-03 14:22:58 -08:00
releaseJson=$(curl -s "${versionsUrl}")
if [[ "$requestedVersion" == "latest" ]]; then
2017-01-10 20:19:39 -08:00
pre=$([[ "${prerelease}" == "true" ]] && echo "null" || echo "-pre")
version=$(echo "${releaseJson}" | python3 -c "import json,sys,collections;obj=json.load(sys.stdin, object_pairs_hook=collections.OrderedDict);latest=list(v for v in obj if '${pre}' not in v)[-1];print(latest)")
2017-01-03 14:22:58 -08:00
else
version="${requestedVersion}"
2016-10-25 13:05:24 -07:00
fi
2017-01-03 14:22:58 -08:00
if ! sourceTarballUrl=$(echo "${releaseJson}" | python3 -c 'import json,sys;obj=json.load(sys.stdin);print(obj[sys.argv[1]]["sourceTarballUrl"])' "${version}"); then
2016-12-30 13:12:22 -08:00
echo "No source code for version ${requestedVersion}"
2016-10-25 13:05:24 -07:00
exit 1
fi
2016-12-30 13:12:22 -08:00
# Build data
if [[ -z "${dataJson}" ]]; then
if [[ -z "${restoreUrl}" ]]; then
data=$(cat <<EOF
{
"boxVersionsUrl": "${versionsUrl}",
"fqdn": "${domain}",
"provider": "${provider}",
2017-01-11 12:35:31 -08:00
"apiServerOrigin": "${apiServerOrigin}",
2016-12-30 13:12:22 -08:00
"tlsConfig": {
"provider": "${tlsProvider}"
},
"dnsConfig": {
"provider": "${dnsProvider}"
},
"backupConfig" : {
"provider": "filesystem",
"backupFolder": "/var/backups",
"key": "${encryptionKey}"
},
2017-01-10 20:19:39 -08:00
"updateConfig": {
"prerelease": ${prerelease}
},
2016-12-30 13:12:22 -08:00
"version": "${version}"
}
EOF
)
else
data=$(cat <<EOF
{
"boxVersionsUrl": "${versionsUrl}",
"fqdn": "${domain}",
"provider": "${provider}",
2017-01-11 12:35:31 -08:00
"apiServerOrigin": "${apiServerOrigin}",
2016-12-30 13:12:22 -08:00
"restore": {
"url": "${restoreUrl}",
"key": "${encryptionKey}"
},
"version": "${version}"
}
EOF
)
fi
else
data="${dataJson}"
fi
2017-01-03 14:48:37 -08:00
echo "=> Downloading version ${version} ..."
2016-12-23 18:28:18 -08:00
box_src_tmp_dir=$(mktemp -dt box-src-XXXXXX)
2016-12-26 11:57:14 -08:00
if ! curl -sL "${sourceTarballUrl}" | tar -zxf - -C "${box_src_tmp_dir}"; then
2016-12-23 18:28:18 -08:00
echo "Could not download source tarball. See ${LOG_FILE} for details"
exit 1
fi
2017-01-03 14:48:37 -08:00
if [[ "${initBaseImage}" == "true" ]]; then
2017-01-09 12:57:34 -08:00
echo "=> Installing base dependencies and downloading docker images (this takes some time) ..."
2017-01-09 09:22:22 -08:00
if ! /bin/bash "${box_src_tmp_dir}/baseimage/initializeBaseUbuntuImage.sh" "${provider}" "../src" &>> "${LOG_FILE}"; then
2017-01-03 14:48:37 -08:00
echo "Init script failed. See ${LOG_FILE} for details"
exit 1
fi
fi
echo "=> Installing version ${version} (this takes some time) ..."
2016-12-26 21:15:16 -08:00
if ! /bin/bash "${box_src_tmp_dir}/scripts/installer.sh" --data "${data}" &>> "${LOG_FILE}"; then
2016-10-25 12:47:51 -07:00
echo "Failed to install cloudron. See ${LOG_FILE} for details"
exit 1
fi
2016-10-25 11:27:58 +02:00
2016-12-30 11:35:03 +01:00
echo -n "=> Waiting for cloudron to be ready (this takes some time) ..."
2016-10-24 14:52:30 -07:00
while true; do
echo -n "."
2017-01-09 12:57:34 -08:00
if status=$(curl -q -f "http://localhost:3000/api/v1/cloudron/status" 2>/dev/null); then
[[ -z "$domain" ]] && break # with no domain, we are up and running
[[ "$status" == *"\"tls\": true"* ]] && break # with a domain, wait for the cert
2016-10-24 14:52:30 -07:00
fi
sleep 10
done
2017-01-05 20:18:57 -08:00
echo -e "\n\nRebooting this server now to let bootloader changes take effect.\n"
if [[ -n "${domain}" ]]; then
echo -e "Visit https://my.${domain} to finish setup once the server has rebooted.\n"
else
echo -e "Visit https://<IP> to finish setup once the server has rebooted.\n"
fi
2016-12-27 15:20:26 -08:00
if [[ "${initBaseImage}" == "true" ]]; then
systemctl reboot
fi