2015-11-25 21:43:59 +00:00
#!/usr/bin/env bash
2017-07-26 17:00:08 +00:00
# shellcheck disable=SC1090
2015-10-28 22:29:34 +00:00
# Pi-hole: A black hole for Internet advertisements
2022-06-19 00:17:10 +00:00
# (c) Pi-hole (https://pi-hole.net)
2017-02-22 17:55:20 +00:00
# Network-wide ad blocking via your own hardware.
#
2017-07-26 17:00:08 +00:00
# Installs and Updates Pi-hole
2015-12-06 14:31:49 +00:00
#
2017-02-22 17:55:20 +00:00
# This file is copyright under the latest version of the EUPL.
# Please see LICENSE file for your rights under this license.
2015-10-28 22:29:34 +00:00
# pi-hole.net/donate
2015-06-14 03:14:21 +00:00
#
2017-07-07 00:25:56 +00:00
# Install with this command (from your Linux machine):
2015-06-14 03:14:21 +00:00
#
2018-01-20 13:29:38 +00:00
# curl -sSL https://install.pi-hole.net | bash
2015-06-14 03:14:21 +00:00
2017-07-07 00:25:56 +00:00
# -e option instructs bash to immediately exit if any command [1] has a non-zero exit status
# We do not want users to end up with a partially working install, so we exit the script
# instead of continuing the installation with something broken
2016-10-10 09:24:03 +00:00
set -e
2017-07-07 00:25:56 +00:00
2021-03-18 01:35:27 +00:00
# Append common folders to the PATH to ensure that all basic commands are available.
# When using "su" an incomplete PATH could be passed: https://github.com/pi-hole/pi-hole/issues/3209
2020-07-02 21:19:30 +00:00
export PATH += ':/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin'
2015-10-28 22:29:34 +00:00
######## VARIABLES #########
2017-07-07 00:25:56 +00:00
# For better maintainability, we store as much information that can change in variables
2018-02-07 05:45:23 +00:00
# This allows us to make a change in one place that can propagate to all instances of the variable
2017-07-07 00:25:56 +00:00
# These variables should all be GLOBAL variables, written in CAPS
# Local variables will be in lowercase and will exist only within functions
# It's still a work in progress, so you may see some variance in this guideline until it is complete
2022-06-19 00:17:10 +00:00
# Dialog result codes
# dialog code values can be set by environment variables, we only override if
# the env var is not set or empty.
: " ${ DIALOG_OK : =0 } "
: " ${ DIALOG_CANCEL : =1 } "
: " ${ DIALOG_ESC : =255 } "
2019-01-13 14:51:30 +00:00
# List of supported DNS servers
DNS_SERVERS = $( cat << EOM
2021-10-04 03:16:19 +00:00
Google ( ECS, DNSSEC) ; 8.8.8.8; 8.8.4.4; 2001:4860:4860:0:0:0:0:8888; 2001:4860:4860:0:0:0:0:8844
2020-11-24 20:02:53 +00:00
OpenDNS ( ECS, DNSSEC) ; 208.67.222.222; 208.67.220.220; 2620:119:35::35; 2620:119:53::53
2019-01-13 14:51:30 +00:00
Level3; 4.2.2.1; 4.2.2.2; ;
Comodo; 8.26.56.26; 8.20.247.20; ;
2021-10-04 03:16:19 +00:00
DNS.WATCH ( DNSSEC) ; 84.200.69.80; 84.200.70.40; 2001:1608:10:25:0:0:1c04:b12f; 2001:1608:10:25:0:0:9249:d69b
2019-01-13 14:51:30 +00:00
Quad9 ( filtered, DNSSEC) ; 9.9.9.9; 149.112.112.112; 2620:fe::fe; 2620:fe::9
Quad9 ( unfiltered, no DNSSEC) ; 9.9.9.10; 149.112.112.10; 2620:fe::10; 2620:fe::fe:10
2021-10-04 03:16:19 +00:00
Quad9 ( filtered, ECS, DNSSEC) ; 9.9.9.11; 149.112.112.11; 2620:fe::11; 2620:fe::fe:11
Cloudflare ( DNSSEC) ; 1.1.1.1; 1.0.0.1; 2606:4700:4700::1111; 2606:4700:4700::1001
2019-01-13 14:51:30 +00:00
EOM
)
2018-01-20 13:55:48 +00:00
# Location for final installation log storage
2021-10-05 15:40:12 +00:00
installLogLoc = "/etc/pihole/install.log"
2017-07-07 00:25:56 +00:00
# This is an important file as it contains information specific to the machine it's being installed on
2021-10-05 15:40:12 +00:00
setupVars = "/etc/pihole/setupVars.conf"
2017-07-07 00:25:56 +00:00
# Pi-hole uses lighttpd as a Web server, and this is the config file for it
2021-10-05 15:40:12 +00:00
lighttpdConfig = "/etc/lighttpd/lighttpd.conf"
2017-07-07 00:25:56 +00:00
# This is a file used for the colorized output
2021-10-05 15:40:12 +00:00
coltable = "/opt/pihole/COL_TABLE"
2015-10-31 14:15:57 +00:00
2019-02-05 09:30:43 +00:00
# Root of the web server
webroot = "/var/www/html"
2021-03-18 01:35:27 +00:00
# We clone (or update) two git repositories during the install. This helps to make sure that we always have the latest versions of the relevant files.
# AdminLTE is used to set up the Web admin interface.
# Pi-hole contains various setup scripts and files which are critical to the installation.
# Search for "PI_HOLE_LOCAL_REPO" in this file to see all such scripts.
# Two notable scripts are gravity.sh (used to generate the HOSTS file) and advanced/Scripts/webpage.sh (used to install the Web admin interface)
2016-01-24 03:31:12 +00:00
webInterfaceGitUrl = "https://github.com/pi-hole/AdminLTE.git"
2019-02-05 09:30:43 +00:00
webInterfaceDir = " ${ webroot } /admin "
2016-01-24 03:31:12 +00:00
piholeGitUrl = "https://github.com/pi-hole/pi-hole.git"
2016-11-02 07:14:25 +00:00
PI_HOLE_LOCAL_REPO = "/etc/.pihole"
2021-03-18 01:35:27 +00:00
# List of pihole scripts, stored in an array
2017-01-15 04:16:27 +00:00
PI_HOLE_FILES = ( chronometer list piholeDebug piholeLogFlush setupLCD update version gravity uninstall webpage)
2018-07-27 16:35:39 +00:00
# This directory is where the Pi-hole scripts will be installed
2017-01-09 07:01:18 +00:00
PI_HOLE_INSTALL_DIR = "/opt/pihole"
2018-07-30 21:26:07 +00:00
PI_HOLE_CONFIG_DIR = "/etc/pihole"
2019-08-24 11:33:32 +00:00
PI_HOLE_BIN_DIR = "/usr/local/bin"
2019-02-05 09:30:43 +00:00
PI_HOLE_BLOCKPAGE_DIR = " ${ webroot } /pihole "
2020-11-16 22:15:02 +00:00
if [ -z " $useUpdateVars " ] ; then
2021-11-25 06:41:40 +00:00
useUpdateVars = false
2020-11-16 22:15:02 +00:00
fi
2016-01-23 00:13:16 +00:00
2018-04-17 08:50:25 +00:00
adlistFile = "/etc/pihole/adlists.list"
2021-03-18 01:35:27 +00:00
# Pi-hole needs an IP address; to begin, these variables are empty since we don't know what the IP is until this script can run
2020-07-15 22:38:08 +00:00
IPV4_ADDRESS = ${ IPV4_ADDRESS }
IPV6_ADDRESS = ${ IPV6_ADDRESS }
2021-03-18 01:35:27 +00:00
# Give settings their default values. These may be changed by prompts later in the script.
2016-11-01 16:46:54 +00:00
QUERY_LOGGING = true
2018-04-15 01:08:16 +00:00
INSTALL_WEB_INTERFACE = true
2018-08-20 23:04:58 +00:00
PRIVACY_LEVEL = 0
2020-09-17 21:13:40 +00:00
CACHE_SIZE = 10000
2017-01-28 14:38:54 +00:00
2018-06-01 08:20:40 +00:00
if [ -z " ${ USER } " ] ; then
2021-11-25 06:41:40 +00:00
USER = " $( id -un) "
2018-06-01 08:20:40 +00:00
fi
2017-01-28 14:38:54 +00:00
2022-07-02 19:36:16 +00:00
# dialog dimensions: Let dialog handle appropriate sizing.
Hardcode whiptail dimensions to 20 rows and 70 chars width
With the suggested way to call the installer via "curl -sSL https://install.pi-hole.net | bash", STDIN is no terminal, but overridden by the curl output, hence in most cases, the minimum dimensions were applied, even on larger screens. All whiptail calls are hence assured to work fine with those dimensions, aside of one case, making the calculations obsolete.
This commit hardcodes the whiptail dimensions to the prior minimum and removes the calculations. This also helps with testing, as it does not matter anymore how the script is called, and developers have a clearly defined space to make dialogs look nice, including line breaks, menu and list heights.
The only case which does not fit the 70 character width, the second menu entry of the "pihole -r" dialog, has been shortened accordingly. This was not an issue before, as "pihole -r" does not override the scripts STDIN and hence did allow larger dimensions based on the now removed calculations.
See the following discussions for reference:
- https://github.com/pi-hole/pi-hole/issues/3323
- https://github.com/pi-hole/pi-hole/pull/4197#issuecomment-876702380
Signed-off-by: MichaIng <micha@dietpi.com>
2021-07-16 17:58:49 +00:00
r = 20
c = 70
2015-10-28 22:29:34 +00:00
2016-10-15 19:52:20 +00:00
######## Undocumented Flags. Shhh ########
2017-07-07 00:25:56 +00:00
# These are undocumented flags; some of which we can use when repairing an installation
# The runUnattended flag is one example of this
2016-10-15 20:12:13 +00:00
reconfigure = false
2016-10-15 19:43:03 +00:00
runUnattended = false
2018-04-15 01:08:16 +00:00
INSTALL_WEB_SERVER = true
# Check arguments for the undocumented flags
for var in " $@ " ; do
2018-07-06 04:48:36 +00:00
case " $var " in
2018-07-26 23:27:06 +00:00
"--reconfigure" ) reconfigure = true; ;
2018-07-06 04:48:36 +00:00
"--unattended" ) runUnattended = true; ;
"--disable-install-webserver" ) INSTALL_WEB_SERVER = false; ;
esac
2018-04-15 01:08:16 +00:00
done
2016-10-15 19:43:03 +00:00
2017-07-07 00:25:56 +00:00
# If the color table file exists,
2017-07-26 17:00:08 +00:00
if [ [ -f " ${ coltable } " ] ] ; then
2018-07-06 04:48:36 +00:00
# source it
2019-05-10 14:13:23 +00:00
source " ${ coltable } "
2018-02-07 05:45:23 +00:00
# Otherwise,
2017-06-21 11:49:05 +00:00
else
2018-07-06 04:48:36 +00:00
# Set these values so the installer can still run in color
COL_NC = '\e[0m' # No Color
COL_LIGHT_GREEN = '\e[1;32m'
COL_LIGHT_RED = '\e[1;31m'
TICK = " [ ${ COL_LIGHT_GREEN } ✓ ${ COL_NC } ] "
CROSS = " [ ${ COL_LIGHT_RED } ✗ ${ COL_NC } ] "
INFO = "[i]"
# shellcheck disable=SC2034
DONE = " ${ COL_LIGHT_GREEN } done! ${ COL_NC } "
OVER = "\\r\\033[K"
2017-06-21 11:49:05 +00:00
fi
2017-07-07 00:25:56 +00:00
# A simple function that just echoes out our logo in ASCII format
# This lets users know that it is a Pi-hole, LLC product
2017-03-09 03:14:21 +00:00
show_ascii_berry( ) {
2021-11-25 06:41:40 +00:00
echo -e "
2017-06-21 11:49:05 +00:00
${ COL_LIGHT_GREEN } .; ; ,.
2017-03-09 03:14:21 +00:00
.ccccc:,.
:cccclll:. ..,,
:ccccclll. ; ooodc
' ccll:; ll .oooodc
.; cll.; ; looo:.
2017-06-21 11:49:05 +00:00
${ COL_LIGHT_RED } .. ',' .
2017-03-09 03:14:21 +00:00
.',,,,,,' .
.' ,,,,,,,,,,.
.' ,,,,,,,,,,,,....
....'' ',,,,,,,' .......
......... .... .........
.......... ..........
.......... ..........
......... .... .........
........,,,,,,,' ......
....' ,,,,,,,,,,,,.
.',,,,,,,,,' .
.',,,,,,' .
2017-06-21 11:49:05 +00:00
..'' ' .${ COL_NC }
2017-03-09 03:14:21 +00:00
"
}
2018-11-01 19:00:08 +00:00
is_command( ) {
2021-03-18 01:35:27 +00:00
# Checks to see if the given command (passed as a string argument) exists on the system.
# The function returns 0 (success) if the command exists, and 1 if it doesn't.
2018-11-01 19:00:08 +00:00
local check_command = " $1 "
command -v " ${ check_command } " >/dev/null 2>& 1
}
2020-05-25 16:40:20 +00:00
os_check( ) {
2020-07-21 17:47:13 +00:00
if [ " $PIHOLE_SKIP_OS_CHECK " != true ] ; then
# This function gets a list of supported OS versions from a TXT record at versions.pi-hole.net
# and determines whether or not the script is running on one of those systems
2020-08-23 09:48:15 +00:00
local remote_os_domain valid_os valid_version valid_response detected_os detected_version display_warning cmdResult digReturnCode response
2021-04-15 13:59:10 +00:00
remote_os_domain = ${ OS_CHECK_DOMAIN_NAME :- "versions.pi-hole.net" }
2020-07-21 17:47:13 +00:00
2021-12-02 13:44:50 +00:00
detected_os = $( grep '^ID=' /etc/os-release | cut -d '=' -f2 | tr -d '"' )
2020-08-14 18:42:54 +00:00
detected_version = $( grep VERSION_ID /etc/os-release | cut -d '=' -f2 | tr -d '"' )
2020-07-21 17:47:13 +00:00
2021-04-15 13:59:10 +00:00
cmdResult = " $( dig +short -t txt " ${ remote_os_domain } " @ns1.pi-hole.net 2>& 1; echo $? ) "
2021-03-18 01:35:27 +00:00
# Gets the return code of the previous command (last line)
2020-08-23 09:48:15 +00:00
digReturnCode = " ${ cmdResult ##* $'\n' } "
2020-07-21 17:47:13 +00:00
2020-08-23 09:48:15 +00:00
if [ ! " ${ digReturnCode } " = = "0" ] ; then
valid_response = false
2020-07-27 15:01:55 +00:00
else
2021-03-18 01:35:27 +00:00
# Dig returned 0 (success), so get the actual response, and loop through it to determine if the detected variables above are valid
2020-08-23 09:48:15 +00:00
response = " ${ cmdResult %% $'\n' * } "
2021-03-18 01:35:27 +00:00
# If the value of ${response} is a single 0, then this is the return code, not an actual response.
2020-08-23 09:48:15 +00:00
if [ " ${ response } " = = 0 ] ; then
valid_response = false
fi
IFS = " " read -r -a supportedOS < <( echo " ${ response } " | tr -d '"' )
2020-08-14 21:37:58 +00:00
for distro_and_versions in " ${ supportedOS [@] } "
do
distro_part = " ${ distro_and_versions %%=* } "
versions_part = " ${ distro_and_versions ##*= } "
2022-07-02 13:00:17 +00:00
# If the distro part is a (case-insensitive) substring of the computer OS
2020-08-14 21:37:58 +00:00
if [ [ " ${ detected_os ^^ } " = ~ ${ distro_part ^^ } ] ] ; then
valid_os = true
IFS = "," read -r -a supportedVer <<< " ${ versions_part } "
for version in " ${ supportedVer [@] } "
do
if [ [ " ${ detected_version } " = ~ $version ] ] ; then
valid_version = true
break
fi
done
2020-07-27 15:01:55 +00:00
break
2020-08-14 21:37:58 +00:00
fi
2020-07-21 17:47:13 +00:00
done
2020-08-14 21:37:58 +00:00
fi
2020-05-25 16:40:20 +00:00
2020-08-23 09:48:15 +00:00
if [ " $valid_os " = true ] && [ " $valid_version " = true ] && [ ! " $valid_response " = false ] ; then
2020-07-21 17:47:13 +00:00
display_warning = false
fi
2020-05-25 16:40:20 +00:00
2020-08-14 18:42:54 +00:00
if [ " $display_warning " != false ] ; then
2020-08-23 09:48:15 +00:00
if [ " $valid_response " = false ] ; then
if [ " ${ digReturnCode } " -eq 0 ] ; then
2020-08-23 13:50:04 +00:00
errStr = "dig succeeded, but response was blank. Please contact support"
2020-08-23 09:48:15 +00:00
else
errStr = " dig failed with return code ${ digReturnCode } "
fi
printf " %b %bRetrieval of supported OS list failed. %s. %b\\n" " ${ CROSS } " " ${ COL_LIGHT_RED } " " ${ errStr } " " ${ COL_NC } "
printf " %bUnable to determine if the detected OS (%s %s) is supported%b\\n" " ${ COL_LIGHT_RED } " " ${ detected_os ^ } " " ${ detected_version } " " ${ COL_NC } "
2020-08-23 13:50:04 +00:00
printf " Possible causes for this include:\\n"
printf " - Firewall blocking certain DNS lookups from Pi-hole device\\n"
printf " - ns1.pi-hole.net being blocked (required to obtain TXT record from versions.pi-hole.net containing supported operating systems)\\n"
printf " - Other internet connectivity issues\\n"
2020-08-23 09:48:15 +00:00
else
printf " %b %bUnsupported OS detected: %s %s%b\\n" " ${ CROSS } " " ${ COL_LIGHT_RED } " " ${ detected_os ^ } " " ${ detected_version } " " ${ COL_NC } "
printf " If you are seeing this message and you do have a supported OS, please contact support.\\n"
fi
printf "\\n"
2022-07-02 13:00:17 +00:00
printf " %bhttps://docs.pi-hole.net/main/prerequisites/#supported-operating-systems%b\\n" " ${ COL_LIGHT_GREEN } " " ${ COL_NC } "
2020-08-23 09:48:15 +00:00
printf "\\n"
printf " If you wish to attempt to continue anyway, you can try one of the following commands to skip this check:\\n"
2020-07-21 17:47:13 +00:00
printf "\\n"
2020-07-22 21:30:51 +00:00
printf " e.g: If you are seeing this message on a fresh install, you can run:\\n"
2022-04-13 19:30:12 +00:00
printf " %bcurl -sSL https://install.pi-hole.net | sudo PIHOLE_SKIP_OS_CHECK=true bash%b\\n" " ${ COL_LIGHT_GREEN } " " ${ COL_NC } "
2020-07-22 21:29:38 +00:00
printf "\\n"
printf " If you are seeing this message after having run pihole -up:\\n"
2022-04-13 19:30:12 +00:00
printf " %bsudo PIHOLE_SKIP_OS_CHECK=true pihole -r%b\\n" " ${ COL_LIGHT_GREEN } " " ${ COL_NC } "
2020-07-22 21:29:38 +00:00
printf " (In this case, your previous run of pihole -up will have already updated the local repository)\\n"
printf "\\n"
2020-08-23 09:48:15 +00:00
printf " It is possible that the installation will still fail at this stage due to an unsupported configuration.\\n"
2020-07-21 17:47:13 +00:00
printf " If that is the case, you can feel free to ask the community on Discourse with the %bCommunity Help%b category:\\n" " ${ COL_LIGHT_RED } " " ${ COL_NC } "
2020-08-23 09:48:15 +00:00
printf " %bhttps://discourse.pi-hole.net/c/bugs-problems-issues/community-help/%b\\n" " ${ COL_LIGHT_GREEN } " " ${ COL_NC } "
printf "\\n"
2020-07-21 17:47:13 +00:00
exit 1
else
printf " %b %bSupported OS detected%b\\n" " ${ TICK } " " ${ COL_LIGHT_GREEN } " " ${ COL_NC } "
fi
2020-05-25 16:40:20 +00:00
else
2020-07-21 17:47:13 +00:00
printf " %b %bPIHOLE_SKIP_OS_CHECK env variable set to true - installer will continue%b\\n" " ${ INFO } " " ${ COL_LIGHT_GREEN } " " ${ COL_NC } "
2020-05-25 16:40:20 +00:00
fi
}
2022-04-04 14:38:30 +00:00
# This function waits for dpkg to unlock, which signals that the previous apt-get command has finished.
test_dpkg_lock( ) {
i = 0
2022-04-07 07:11:53 +00:00
printf " %b Waiting for package manager to finish (up to 30 seconds)\\n" " ${ INFO } "
2022-04-04 14:38:30 +00:00
# fuser is a program to show which processes use the named files, sockets, or filesystems
# So while the lock is held,
while fuser /var/lib/dpkg/lock >/dev/null 2>& 1
do
# we wait half a second,
sleep 0.5
# increase the iterator,
( ( i = i+1) )
# exit if waiting for more then 30 seconds
if [ [ $i -gt 60 ] ] ; then
2022-04-04 14:48:34 +00:00
printf " %b %bError: Could not verify package manager finished and released lock. %b\\n" " ${ CROSS } " " ${ COL_LIGHT_RED } " " ${ COL_NC } "
printf " Attempt to install packages manually and retry.\\n"
2022-04-04 14:38:30 +00:00
exit 1;
fi
done
# and then report success once dpkg is unlocked.
return 0
}
2016-08-19 21:31:11 +00:00
# Compatibility
2021-08-03 20:48:26 +00:00
package_manager_detect( ) {
2022-07-04 19:35:08 +00:00
# TODO - pull common packages for both distributions out into a common variable, then add
# the distro-specific ones below.
2021-11-25 06:41:40 +00:00
# First check to see if apt-get is installed.
if is_command apt-get ; then
# Set some global variables here
# We don't set them earlier since the installed package manager might be rpm, so these values would be different
PKG_MANAGER = "apt-get"
# A variable to store the command used to update the package cache
UPDATE_PKG_CACHE = " ${ PKG_MANAGER } update "
# The command we will use to actually install packages
PKG_INSTALL = ( " ${ PKG_MANAGER } " -qq --no-install-recommends install)
# grep -c will return 1 if there are no matches. This is an acceptable condition, so we OR TRUE to prevent set -e exiting the script.
PKG_COUNT = " ${ PKG_MANAGER } -s -o Debug::NoLocking=true upgrade | grep -c ^Inst || true "
# Update package cache
update_package_cache || exit 1
# Check for and determine version number (major and minor) of current php install
2022-06-13 22:58:10 +00:00
local phpVer = "php"
2021-11-25 06:41:40 +00:00
if is_command php ; then
2022-06-14 21:04:31 +00:00
phpVer = " $( php <<< "<?php echo PHP_VERSION ?>" ) "
2022-06-13 22:05:27 +00:00
# Check if the first character of the string is numeric
if [ [ ${ phpVer : 0 : 1 } = ~ [ 1-9] ] ] ; then
printf " %b Existing PHP installation detected : PHP version %s\\n" " ${ INFO } " " ${ phpVer } "
printf -v phpInsMajor "%d" " $( php <<< "<?php echo PHP_MAJOR_VERSION ?>" ) "
printf -v phpInsMinor "%d" " $( php <<< "<?php echo PHP_MINOR_VERSION ?>" ) "
phpVer = " php $phpInsMajor . $phpInsMinor "
else
printf " %b No valid PHP installation detected!\\n" " ${ CROSS } "
printf " %b PHP version : %s\\n" " ${ INFO } " " ${ phpVer } "
printf " %b Aborting installation.\\n" " ${ CROSS } "
exit 1
2022-06-13 13:06:15 +00:00
fi
2021-11-25 06:41:40 +00:00
fi
2022-07-02 13:00:17 +00:00
# Packages required to perform the os_check (stored as an array)
2021-11-25 06:41:40 +00:00
OS_CHECK_DEPS = ( grep dnsutils)
# Packages required to run this install script (stored as an array)
2022-06-19 00:17:10 +00:00
INSTALLER_DEPS = ( git iproute2 dialog ca-certificates)
2021-11-25 06:41:40 +00:00
# Packages required to run Pi-hole (stored as an array)
2022-03-01 08:07:51 +00:00
PIHOLE_DEPS = ( cron curl iputils-ping psmisc sudo unzip idn2 libcap2-bin dns-root-data libcap2 netcat-openbsd procps)
2021-11-25 06:41:40 +00:00
# Packages required for the Web admin interface (stored as an array)
# It's useful to separate this from Pi-hole, since the two repos are also setup separately
PIHOLE_WEB_DEPS = ( lighttpd " ${ phpVer } -common " " ${ phpVer } -cgi " " ${ phpVer } -sqlite3 " " ${ phpVer } -xml " " ${ phpVer } -intl " )
# Prior to PHP8.0, JSON functionality is provided as dedicated module, required by Pi-hole AdminLTE: https://www.php.net/manual/json.installation.php
if [ [ -z " ${ phpInsMajor } " || " ${ phpInsMajor } " -lt 8 ] ] ; then
PIHOLE_WEB_DEPS += ( " ${ phpVer } -json " )
fi
# The Web server user,
LIGHTTPD_USER = "www-data"
# group,
LIGHTTPD_GROUP = "www-data"
# and config file
LIGHTTPD_CFG = "lighttpd.conf.debian"
# If apt-get is not found, check for rpm.
elif is_command rpm ; then
# Then check if dnf or yum is the package manager
if is_command dnf ; then
PKG_MANAGER = "dnf"
else
PKG_MANAGER = "yum"
fi
2017-03-03 10:28:35 +00:00
2021-11-25 06:41:40 +00:00
# These variable names match the ones for apt-get. See above for an explanation of what they are for.
PKG_INSTALL = ( " ${ PKG_MANAGER } " install -y)
2022-07-04 21:00:16 +00:00
# CentOS package manager returns 100 when there are packages to update so we need to || true to prevent the script from exiting.
PKG_COUNT = " ${ PKG_MANAGER } check-update | egrep '(.i686|.x86|.noarch|.arm|.src)' | wc -l || true "
2021-11-25 06:41:40 +00:00
OS_CHECK_DEPS = ( grep bind-utils)
2022-07-04 19:35:08 +00:00
INSTALLER_DEPS = ( git dialog iproute newt procps-ng which chkconfig ca-certificates)
2022-02-01 06:38:57 +00:00
PIHOLE_DEPS = ( cronie curl findutils sudo unzip libidn2 psmisc libcap nmap-ncat)
2021-11-25 06:41:40 +00:00
PIHOLE_WEB_DEPS = ( lighttpd lighttpd-fastcgi php-common php-cli php-pdo php-xml php-json php-intl)
LIGHTTPD_USER = "lighttpd"
LIGHTTPD_GROUP = "lighttpd"
LIGHTTPD_CFG = "lighttpd.conf.fedora"
# If neither apt-get or yum/dnf package managers were found
2018-05-01 06:28:55 +00:00
else
2021-11-25 06:41:40 +00:00
# we cannot install required packages
printf " %b No supported package manager found\\n" " ${ CROSS } "
# so exit the installer
exit
2018-07-06 04:48:36 +00:00
fi
2017-01-21 20:34:47 +00:00
}
2015-11-08 23:21:02 +00:00
2021-09-12 20:40:37 +00:00
select_rpm_php( ) {
2021-11-25 06:41:40 +00:00
# If the host OS is Fedora,
if grep -qiE 'fedora|fedberry' /etc/redhat-release; then
# all required packages should be available by default with the latest fedora release
: # continue
# or if host OS is CentOS,
elif grep -qiE 'centos|scientific' /etc/redhat-release; then
# Pi-Hole currently supports CentOS 7+ with PHP7+
SUPPORTED_CENTOS_VERSION = 7
SUPPORTED_CENTOS_PHP_VERSION = 7
# Check current CentOS major release version
CURRENT_CENTOS_VERSION = $( grep -oP '(?<= )[0-9]+(?=\.?)' /etc/redhat-release)
# Check if CentOS version is supported
if [ [ $CURRENT_CENTOS_VERSION -lt $SUPPORTED_CENTOS_VERSION ] ] ; then
printf " %b CentOS %s is not supported.\\n" " ${ CROSS } " " ${ CURRENT_CENTOS_VERSION } "
printf " Please update to CentOS release %s or later.\\n" " ${ SUPPORTED_CENTOS_VERSION } "
# exit the installer
exit
fi
# php-json is not required on CentOS 7 as it is already compiled into php
# verifiy via `php -m | grep json`
if [ [ $CURRENT_CENTOS_VERSION -eq 7 ] ] ; then
# create a temporary array as arrays are not designed for use as mutable data structures
CENTOS7_PIHOLE_WEB_DEPS = ( )
for i in " ${ !PIHOLE_WEB_DEPS[@] } " ; do
if [ [ ${ PIHOLE_WEB_DEPS [i] } != "php-json" ] ] ; then
CENTOS7_PIHOLE_WEB_DEPS += ( " ${ PIHOLE_WEB_DEPS [i] } " )
fi
done
# re-assign the clean dependency array back to PIHOLE_WEB_DEPS
PIHOLE_WEB_DEPS = ( " ${ CENTOS7_PIHOLE_WEB_DEPS [@] } " )
unset CENTOS7_PIHOLE_WEB_DEPS
fi
2022-07-04 21:00:16 +00:00
if rpm -qa | grep -qi 'epel' ; then
printf " %b EPEL repository already installed\\n" " ${ TICK } "
else
# CentOS requires the EPEL repository to gain access to Fedora packages
if [ [ CURRENT_CENTOS_VERSION -eq 7 ] ] ; then
EPEL_PKG = "https://dl.fedoraproject.org/pub/epel/epel-release-latest-7.noarch.rpm"
elif [ [ CURRENT_CENTOS_VERSION -eq 8 ] ] ; then
EPEL_PKG = "https://dl.fedoraproject.org/pub/epel/epel-release-latest-8.noarch.rpm"
fi
printf " %b Enabling EPEL package repository (https://fedoraproject.org/wiki/EPEL)\\n" " ${ INFO } "
" ${ PKG_INSTALL [@] } " ${ EPEL_PKG }
printf " %b Installed %s\\n" " ${ TICK } " " ${ EPEL_PKG } "
2021-11-25 06:41:40 +00:00
fi
2022-07-03 00:47:41 +00:00
2021-11-25 06:41:40 +00:00
# The default php on CentOS 7.x is 5.4 which is EOL
# Check if the version of PHP available via installed repositories is >= to PHP 7
AVAILABLE_PHP_VERSION = $( " ${ PKG_MANAGER } " info php | grep -i version | grep -o '[0-9]\+' | head -1)
if [ [ $AVAILABLE_PHP_VERSION -ge $SUPPORTED_CENTOS_PHP_VERSION ] ] ; then
# Since PHP 7 is available by default, install via default PHP package names
: # do nothing as PHP is current
2022-07-04 21:00:16 +00:00
printf "PHP 7 is installed"
2021-09-12 20:40:37 +00:00
else
2021-11-25 06:41:40 +00:00
REMI_PKG = "remi-release"
REMI_REPO = "remi-php72"
2022-07-04 21:00:16 +00:00
REMI_REPO_URL = " https://rpms.remirepo.net/enterprise/ ${ REMI_PKG } - $( rpm -E '%{rhel}' ) .rpm "
# The PHP version available via default repositories is older than version 7
dialog --no-shadow --keep-tite \
--title "PHP 7 Update (recommended)" \
--defaultno \
--yesno " PHP 7.x is recommended for both security and language features.\
2022-07-02 19:36:16 +00:00
\\ n\\ nWould you like to install PHP7 via Remi' s RPM repository?\
\\ n\\ nSee: https://rpms.remirepo.net for more information" \
2022-07-04 21:00:16 +00:00
" ${ r } " " ${ c } " && result = 0 || result = $?
2022-06-19 00:17:10 +00:00
2022-07-04 21:00:16 +00:00
case ${ result } in
" ${ DIALOG_OK } " )
printf " %b Installing PHP 7 via Remi's RPM repository\\n" " ${ INFO } "
2021-11-25 06:41:40 +00:00
" ${ PKG_INSTALL [@] } " "yum-utils" & > /dev/null
2022-07-04 21:00:16 +00:00
if rpm -q ${ REMI_PKG } & > /dev/null; then
printf " %b Remi's RPM repository is already installed\\n" " ${ TICK } "
else
2022-06-19 00:17:10 +00:00
printf " %b Enabling Remi's RPM repository (https://rpms.remirepo.net)\\n" " ${ INFO } "
2022-07-04 21:00:16 +00:00
yum -y install " ${ REMI_REPO_URL } "
printf " %b Installed %s from %s\\n" " ${ TICK } " " ${ REMI_PKG } " " ${ REMI_REPO_URL } "
2022-06-19 00:17:10 +00:00
printf " %b Remi's RPM repository has been enabled for PHP7\\n" " ${ TICK } "
2022-07-04 21:00:16 +00:00
fi
yum-config-manager --disable 'remi-php*'
yum-config-manager --enable " ${ REMI_REPO } "
2021-11-25 06:41:40 +00:00
# trigger an install/update of PHP to ensure previous version of PHP is updated from REMI
if " ${ PKG_INSTALL [@] } " "php-cli" & > /dev/null; then
printf " %b PHP7 installed/updated via Remi's RPM repository\\n" " ${ TICK } "
else
printf " %b There was a problem updating to PHP7 via Remi's RPM repository\\n" " ${ CROSS } "
exit 1
fi
2022-07-04 21:00:16 +00:00
; ;
2022-06-19 00:17:10 +00:00
# User chose not to install PHP 7 via Remi's RPM repository
" ${ DIALOG_CANCEL } " )
# User decided to NOT update PHP from REMI, attempt to install the default available PHP version
printf " %b User opt-out of PHP 7 upgrade on CentOS. Deprecated PHP may be in use.\\n" " ${ INFO } "
; ;
# User closed the dialog window
" ${ DIALOG_ESC } " )
printf " %b Escape pressed, exiting installer at Remi dialog window\\n" " ${ CROSS } "
2021-11-25 06:41:40 +00:00
exit 1
2022-06-19 00:17:10 +00:00
; ;
esac
2021-09-12 20:40:37 +00:00
fi
2022-06-19 00:17:10 +00:00
2022-07-04 21:00:16 +00:00
else
2022-06-19 00:17:10 +00:00
# Warn user of unsupported version of Fedora or CentOS
2022-07-04 21:00:16 +00:00
dialog --no-shadow --keep-tite \
2022-06-19 00:17:10 +00:00
--title "Unsupported RPM based distribution" \
--defaultno \
--no-button "Exit" \
--yes-button "Continue" \
2022-07-02 19:36:16 +00:00
--yesno " Would you like to continue installation on an unsupported RPM based distribution?\
\\ n\\ nPlease ensure the following packages have been installed manually:\
\\ n\\ n- lighttpd\\ n- lighttpd-fastcgi\\ n- PHP version 7+" \
2022-07-04 21:00:16 +00:00
" ${ r } " " ${ c } " && result = 0 || result = $?
2022-06-19 00:17:10 +00:00
case ${ result } in
# User chose to continue installation on an unsupported RPM based distribution
" ${ DIALOG_OK } " )
printf " %b User opted to continue installation on an unsupported RPM based distribution.\\n" " ${ INFO } "
; ;
# User chose not to continue installation on an unsupported RPM based distribution
" ${ DIALOG_CANCEL } " )
printf " %b User opted not to continue installation on an unsupported RPM based distribution.\\n" " ${ INFO } "
exit 1
; ;
" ${ DIALOG_ESC } " )
printf " %b Escape pressed, exiting installer at unsupported RPM based distribution dialog window\\n" " ${ CROSS } "
exit 1
; ;
esac
2021-09-12 20:40:37 +00:00
fi
}
2018-07-27 16:35:39 +00:00
# A function for checking if a directory is a git repository
2016-11-02 07:14:25 +00:00
is_repo( ) {
2018-07-06 04:48:36 +00:00
# Use a named, local variable instead of the vague $1, which is the first argument passed to this function
# These local variables should always be lowercase
local directory = " ${ 1 } "
# A variable to store the return code
local rc
# If the first argument passed to this function is a directory,
if [ [ -d " ${ directory } " ] ] ; then
# move into the directory
2019-11-08 19:11:55 +00:00
pushd " ${ directory } " & > /dev/null || return 1
2018-07-27 16:35:39 +00:00
# Use git to check if the directory is a repo
2018-07-06 04:48:36 +00:00
# git -C is not used here to support git versions older than 1.8.4
git status --short & > /dev/null || rc = $?
# If the command was not successful,
else
# Set a non-zero return code if directory does not exist
rc = 1
fi
# Move back into the directory the user started in
2019-11-08 19:11:55 +00:00
popd & > /dev/null || return 1
2018-07-06 04:48:36 +00:00
# Return the code; if one is not set, return 0
return " ${ rc :- 0 } "
2016-11-02 07:14:25 +00:00
}
2017-07-07 00:25:56 +00:00
# A function to clone a repo
2016-11-02 07:14:25 +00:00
make_repo( ) {
2018-07-06 04:48:36 +00:00
# Set named variables for better readability
local directory = " ${ 1 } "
local remoteRepo = " ${ 2 } "
2019-10-28 22:35:01 +00:00
2018-07-06 04:48:36 +00:00
# The message to display when this function is running
str = " Clone ${ remoteRepo } into ${ directory } "
# Display the message and use the color table to preface the message with an "info" indicator
2018-10-31 18:14:05 +00:00
printf " %b %s..." " ${ INFO } " " ${ str } "
2018-07-06 04:48:36 +00:00
# If the directory exists,
if [ [ -d " ${ directory } " ] ] ; then
2020-12-02 20:19:59 +00:00
# Return with a 1 to exit the installer. We don't want to overwrite what could already be here in case it is not ours
2020-12-02 20:49:07 +00:00
str = " Unable to clone ${ remoteRepo } into ${ directory } : Directory already exists "
2020-12-02 20:35:40 +00:00
printf "%b %b%s\\n" " ${ OVER } " " ${ CROSS } " " ${ str } "
2020-12-02 20:19:59 +00:00
return 1
2018-07-06 04:48:36 +00:00
fi
# Clone the repo and return the return code from this command
2019-05-07 17:42:11 +00:00
git clone -q --depth 20 " ${ remoteRepo } " " ${ directory } " & > /dev/null || return $?
2019-10-28 22:35:01 +00:00
# Move into the directory that was passed as an argument
2019-11-08 19:11:55 +00:00
pushd " ${ directory } " & > /dev/null || return 1
2020-03-08 23:53:14 +00:00
# Check current branch. If it is master, then reset to the latest available tag.
2020-05-11 21:03:33 +00:00
# In case extra commits have been added after tagging/release (i.e in case of metadata updates/README.MD tweaks)
2019-11-08 19:18:35 +00:00
curBranch = $( git rev-parse --abbrev-ref HEAD)
2021-03-18 01:35:27 +00:00
if [ [ " ${ curBranch } " = = "master" ] ] ; then
# If we're calling make_repo() then it should always be master, we may not need to check.
git reset --hard " $( git describe --abbrev= 0 --tags) " || return $?
2019-10-28 22:35:01 +00:00
fi
2018-07-06 04:48:36 +00:00
# Show a colored message showing it's status
2018-10-31 18:14:05 +00:00
printf "%b %b %s\\n" " ${ OVER } " " ${ TICK } " " ${ str } "
2020-02-14 19:41:43 +00:00
# Data in the repositories is public anyway so we can make it readable by everyone (+r to keep executable permission if already set by git)
chmod -R a+rX " ${ directory } "
2019-10-28 22:35:01 +00:00
# Move back into the original directory
2019-11-08 19:11:55 +00:00
popd & > /dev/null || return 1
2018-07-06 04:48:36 +00:00
return 0
2016-11-02 07:14:25 +00:00
}
2017-07-07 00:25:56 +00:00
# We need to make sure the repos are up-to-date so we can effectively install Clean out the directory if it exists for git to clone into
2016-11-02 07:14:25 +00:00
update_repo( ) {
2018-07-06 04:48:36 +00:00
# Use named, local variables
# As you can see, these are the same variable names used in the last function,
# but since they are local, their scope does not go beyond this function
# This helps prevent the wrong value from being assigned if you were to set the variable as a GLOBAL one
local directory = " ${ 1 } "
2019-10-28 22:35:01 +00:00
local curBranch
2018-07-06 04:48:36 +00:00
# A variable to store the message we want to display;
# Again, it's useful to store these in variables in case we need to reuse or change the message;
# we only need to make one change here
local str = " Update repo in ${ 1 } "
2020-02-16 11:47:42 +00:00
# Move into the directory that was passed as an argument
2019-11-08 19:11:55 +00:00
pushd " ${ directory } " & > /dev/null || return 1
2018-07-06 04:48:36 +00:00
# Let the user know what's happening
2018-10-31 18:14:05 +00:00
printf " %b %s..." " ${ INFO } " " ${ str } "
2018-07-06 04:48:36 +00:00
# Stash any local commits as they conflict with our working code
git stash --all --quiet & > /dev/null || true # Okay for stash failure
git clean --quiet --force -d || true # Okay for already clean directory
# Pull the latest commits
2021-10-06 00:19:28 +00:00
git pull --no-rebase --quiet & > /dev/null || return $?
2020-03-08 23:53:14 +00:00
# Check current branch. If it is master, then reset to the latest available tag.
2020-05-11 21:03:33 +00:00
# In case extra commits have been added after tagging/release (i.e in case of metadata updates/README.MD tweaks)
2019-11-08 19:18:35 +00:00
curBranch = $( git rev-parse --abbrev-ref HEAD)
2019-10-28 22:35:01 +00:00
if [ [ " ${ curBranch } " = = "master" ] ] ; then
2021-11-25 06:41:40 +00:00
git reset --hard " $( git describe --abbrev= 0 --tags) " || return $?
2019-10-28 22:35:01 +00:00
fi
2018-07-06 04:48:36 +00:00
# Show a completion message
2018-10-31 18:14:05 +00:00
printf "%b %b %s\\n" " ${ OVER } " " ${ TICK } " " ${ str } "
2019-05-03 10:27:56 +00:00
# Data in the repositories is public anyway so we can make it readable by everyone (+r to keep executable permission if already set by git)
2019-05-07 08:46:16 +00:00
chmod -R a+rX " ${ directory } "
2018-07-06 04:48:36 +00:00
# Move back into the original directory
2019-11-08 19:11:55 +00:00
popd & > /dev/null || return 1
2018-07-06 04:48:36 +00:00
return 0
2016-11-02 07:14:25 +00:00
}
2021-03-18 01:35:27 +00:00
# A function that combines the previous git functions to update or clone a repo
2016-11-02 07:14:25 +00:00
getGitFiles( ) {
2018-07-06 04:48:36 +00:00
# Setup named variables for the git repos
# We need the directory
local directory = " ${ 1 } "
# as well as the repo URL
local remoteRepo = " ${ 2 } "
# A local variable containing the message to be displayed
local str = " Check for existing repository in ${ 1 } "
# Show the message
2018-10-31 18:14:05 +00:00
printf " %b %s..." " ${ INFO } " " ${ str } "
2018-07-06 04:48:36 +00:00
# Check if the directory is a repository
if is_repo " ${ directory } " ; then
# Show that we're checking it
2018-10-31 18:14:05 +00:00
printf "%b %b %s\\n" " ${ OVER } " " ${ TICK } " " ${ str } "
2018-07-06 04:48:36 +00:00
# Update the repo, returning an error message on failure
2018-10-31 18:14:05 +00:00
update_repo " ${ directory } " || { printf "\\n %b: Could not update local repository. Contact support.%b\\n" " ${ COL_LIGHT_RED } " " ${ COL_NC } " ; exit 1; }
2018-07-06 04:48:36 +00:00
# If it's not a .git repo,
else
# Show an error
2018-10-31 18:14:05 +00:00
printf "%b %b %s\\n" " ${ OVER } " " ${ CROSS } " " ${ str } "
2018-07-06 04:48:36 +00:00
# Attempt to make the repository, showing an error on failure
2018-10-31 18:14:05 +00:00
make_repo " ${ directory } " " ${ remoteRepo } " || { printf "\\n %bError: Could not update local repository. Contact support.%b\\n" " ${ COL_LIGHT_RED } " " ${ COL_NC } " ; exit 1; }
2018-07-06 04:48:36 +00:00
fi
echo ""
2021-03-18 01:35:27 +00:00
# Success via one of the two branches, as the commands would exit if they failed.
2018-07-06 04:48:36 +00:00
return 0
2016-11-02 07:14:25 +00:00
}
2017-07-07 00:25:56 +00:00
# Reset a repo to get rid of any local changed
2017-05-22 21:43:52 +00:00
resetRepo( ) {
2018-07-06 04:48:36 +00:00
# Use named variables for arguments
local directory = " ${ 1 } "
# Move into the directory
2019-11-08 19:11:55 +00:00
pushd " ${ directory } " & > /dev/null || return 1
2018-07-06 04:48:36 +00:00
# Store the message in a variable
str = " Resetting repository within ${ 1 } ... "
# Show the message
2018-10-31 18:14:05 +00:00
printf " %b %s..." " ${ INFO } " " ${ str } "
2018-07-06 04:48:36 +00:00
# Use git to remove the local changes
git reset --hard & > /dev/null || return $?
2019-05-03 10:27:56 +00:00
# Data in the repositories is public anyway so we can make it readable by everyone (+r to keep executable permission if already set by git)
2019-05-07 08:46:16 +00:00
chmod -R a+rX " ${ directory } "
2018-07-06 04:48:36 +00:00
# And show the status
2018-10-31 18:14:05 +00:00
printf "%b %b %s\\n" " ${ OVER } " " ${ TICK } " " ${ str } "
2019-11-08 19:11:55 +00:00
# Return to where we came from
popd & > /dev/null || return 1
2021-03-18 01:35:27 +00:00
# Function succeeded, as "git reset" would have triggered a return earlier if it failed
2018-07-06 04:48:36 +00:00
return 0
2016-11-02 07:14:25 +00:00
}
2016-10-10 12:45:37 +00:00
find_IPv4_information( ) {
2018-11-03 20:20:40 +00:00
# Detects IPv4 address used for communication to WAN addresses.
# Accepts no arguments, returns no values.
2018-07-06 04:48:36 +00:00
# Named, local variables
local route
2018-11-03 20:20:40 +00:00
local IPv4bare
2018-07-06 04:48:36 +00:00
# Find IP used to route to outside world by checking the the route to Google's public DNS server
route = $( ip route get 8.8.8.8)
2018-11-03 20:20:40 +00:00
# Get just the interface IPv4 address
2018-11-03 20:39:43 +00:00
# shellcheck disable=SC2059,SC2086
# disabled as we intentionally want to split on whitespace and have printf populate
# the variable with just the first field.
2018-11-03 20:20:40 +00:00
printf -v IPv4bare " $( printf ${ route #*src } ) "
# Get the default gateway IPv4 address (the way to reach the Internet)
2018-11-03 20:39:43 +00:00
# shellcheck disable=SC2059,SC2086
2018-11-03 20:20:40 +00:00
printf -v IPv4gw " $( printf ${ route #*via } ) "
if ! valid_ip " ${ IPv4bare } " ; then
IPv4bare = "127.0.0.1"
fi
# Append the CIDR notation to the IP address, if valid_ip fails this should return 127.0.0.1/8
2019-04-27 19:39:45 +00:00
IPV4_ADDRESS = $( ip -oneline -family inet address show | grep " ${ IPv4bare } / " | awk '{print $4}' | awk 'END {print}' )
2016-05-01 02:27:38 +00:00
}
2017-07-07 00:25:56 +00:00
# Get available interfaces that are UP
2016-10-10 10:16:22 +00:00
get_available_interfaces( ) {
2018-07-06 04:48:36 +00:00
# There may be more than one so it's all stored in a variable
availableInterfaces = $( ip --oneline link show up | grep -v "lo" | awk '{print $2}' | cut -d':' -f1 | cut -d'@' -f1)
2016-10-10 10:16:22 +00:00
}
2015-11-10 00:33:32 +00:00
2017-07-07 00:25:56 +00:00
# A function for displaying the dialogs the user sees when first running the installer
2016-01-27 06:11:38 +00:00
welcomeDialogs( ) {
2018-07-06 04:48:36 +00:00
# Display the welcome dialog using an appropriately sized window via the calculation conducted earlier in the script
2022-07-04 21:00:16 +00:00
dialog --no-shadow --clear --keep-tite \
2022-06-19 00:17:10 +00:00
--backtitle "Welcome" \
--title "Pi-hole Automated Installer" \
--msgbox "\\n\\nThis installer will transform your device into a network-wide ad blocker!" \
" ${ r } " " ${ c } " \
2022-07-04 21:00:16 +00:00
--and-widget --clear \
2022-06-19 00:17:10 +00:00
--backtitle "Support Pi-hole" \
--title "Open Source Software" \
--msgbox "\\n\\nThe Pi-hole is free, but powered by your donations: https://pi-hole.net/donate/" \
" ${ r } " " ${ c } " \
2022-07-04 21:00:16 +00:00
--and-widget --clear \
2022-06-19 00:17:10 +00:00
--colors \
--backtitle "Initiating network interface" \
--title "Static IP Needed" \
--no-button "Exit" --yes-button "Continue" \
--defaultno \
2022-07-02 19:36:16 +00:00
--yesno " \\n\\nThe Pi-hole is a SERVER so it needs a STATIC IP ADDRESS to function properly.\\n\\n\
\\ Zb\\ Z1IMPORTANT:\\ Zn If you have not already done so, you must ensure that this device has a static IP.\\ n\\ n\
Depending on your operating system, there are many ways to achieve this, through DHCP reservation, or by manually assigning one.\\ n\\ n\
Please continue when the static addressing has been configured." \
2022-07-04 21:00:16 +00:00
" ${ r } " " ${ c } " && result = 0 || result = " $? "
2022-06-19 00:17:10 +00:00
case " ${ result } " in
" ${ DIALOG_CANCEL } " | " ${ DIALOG_ESC } " )
printf " %b Installer exited at static IP message.\\n" " ${ INFO } "
exit 1
; ;
esac
2015-11-08 23:21:02 +00:00
}
2021-03-18 01:35:27 +00:00
# A function that lets the user pick an interface to use with Pi-hole
2016-01-27 06:11:38 +00:00
chooseInterface( ) {
2022-06-19 00:17:10 +00:00
# Turn the available interfaces into a string so it can be used with dialog
local interfacesList
2018-07-06 04:48:36 +00:00
# Number of available interfaces
local interfaceCount
2022-06-19 00:17:10 +00:00
# POSIX compliant way to get the number of elements in an array
interfaceCount = $( printf "%s\n" " ${ availableInterfaces } " | wc -l)
2018-07-06 04:48:36 +00:00
# If there is one interface,
if [ [ " ${ interfaceCount } " -eq 1 ] ] ; then
# Set it as the interface to use since there is no other option
PIHOLE_INTERFACE = " ${ availableInterfaces } "
# Otherwise,
else
2022-06-19 00:17:10 +00:00
# Set status for the first entry to be selected
status = "ON"
2018-07-06 04:48:36 +00:00
# While reading through the available interfaces
2022-06-19 00:17:10 +00:00
for interface in ${ availableInterfaces } ; do
# Put all these interfaces into a string
interfacesList = " ${ interfacesList } ${ interface } available ${ status } "
# All further interfaces are deselected
status = "OFF"
2018-07-06 04:48:36 +00:00
done
2022-06-19 00:17:10 +00:00
# shellcheck disable=SC2086
# Disable check for double quote here as we are passing a string with spaces
2022-07-04 21:00:16 +00:00
PIHOLE_INTERFACE = $( dialog --no-shadow --keep-tite --output-fd 1 \
2022-07-05 21:23:59 +00:00
--cancel-label "Exit" --ok-label "Select" \
2022-06-19 00:17:10 +00:00
--radiolist "Choose An Interface (press space to toggle selection)" \
${ r } ${ c } " ${ interfaceCount } " ${ interfacesList } )
result = $?
case ${ result } in
" ${ DIALOG_CANCEL } " | " ${ DIALOG_ESC } " )
# Show an error message and exit
printf " %b %s\\n" " ${ CROSS } " "No interface selected, exiting installer"
exit 1
; ;
esac
printf " %b Using interface: %s\\n" " ${ INFO } " " ${ PIHOLE_INTERFACE } "
2018-07-06 04:48:36 +00:00
fi
2016-10-10 12:45:37 +00:00
}
2016-03-06 05:16:23 +00:00
2017-07-07 00:25:56 +00:00
# This lets us prefer ULA addresses over GUA
# This caused problems for some users when their ISP changed their IPv6 addresses
2017-06-02 21:01:48 +00:00
# See https://github.com/pi-hole/pi-hole/issues/1473#issuecomment-301745953
testIPv6( ) {
2018-07-06 04:48:36 +00:00
# first will contain fda2 (ULA)
2018-11-08 19:10:09 +00:00
printf -v first "%s" " ${ 1 %% : * } "
2021-03-18 01:35:27 +00:00
# value1 will contain 253 which is the decimal value corresponding to 0xFD
2018-07-06 04:48:36 +00:00
value1 = $(( ( 0 x$first ) / 256 ))
2021-03-18 01:35:27 +00:00
# value2 will contain 162 which is the decimal value corresponding to 0xA2
2018-07-06 04:48:36 +00:00
value2 = $(( ( 0 x$first ) % 256 ))
# the ULA test is testing for fc00::/7 according to RFC 4193
if ( ( ( value1& 254) = = 252 ) ) ; then
2018-10-31 18:14:05 +00:00
# echoing result to calling function as return value
2018-07-06 04:48:36 +00:00
echo "ULA"
fi
# the GUA test is testing for 2000::/3 according to RFC 4291
if ( ( ( value1& 112) = = 32 ) ) ; then
2018-10-31 18:14:05 +00:00
# echoing result to calling function as return value
2018-07-06 04:48:36 +00:00
echo "GUA"
fi
# the LL test is testing for fe80::/10 according to RFC 4193
if ( ( ( value1) = = 254 ) ) && ( ( ( value2& 192) = = 128 ) ) ; then
2018-10-31 18:14:05 +00:00
# echoing result to calling function as return value
2018-07-06 04:48:36 +00:00
echo "Link-local"
fi
2017-06-02 21:01:48 +00:00
}
2021-10-02 21:29:23 +00:00
find_IPv6_information( ) {
# Detects IPv6 address used for communication to WAN addresses.
2018-07-06 04:48:36 +00:00
IPV6_ADDRESSES = ( $( ip -6 address | grep 'scope global' | awk '{print $2}' ) )
# For each address in the array above, determine the type of IPv6 address it is
for i in " ${ IPV6_ADDRESSES [@] } " ; do
# Check if it's ULA, GUA, or LL by using the function created earlier
result = $( testIPv6 " $i " )
# If it's a ULA address, use it and store it as a global variable
[ [ " ${ result } " = = "ULA" ] ] && ULA_ADDRESS = " ${ i %/* } "
2021-03-18 01:35:27 +00:00
# If it's a GUA address, use it and store it as a global variable
2018-07-06 04:48:36 +00:00
[ [ " ${ result } " = = "GUA" ] ] && GUA_ADDRESS = " ${ i %/* } "
2021-03-18 01:35:27 +00:00
# Else if it's a Link-local address, we cannot use it, so just continue
2018-07-06 04:48:36 +00:00
done
# Determine which address to be used: Prefer ULA over GUA or don't use any if none found
# If the ULA_ADDRESS contains a value,
if [ [ ! -z " ${ ULA_ADDRESS } " ] ] ; then
# set the IPv6 address to the ULA address
IPV6_ADDRESS = " ${ ULA_ADDRESS } "
# Show this info to the user
2021-10-02 21:29:23 +00:00
printf " %b Found IPv6 ULA address\\n" " ${ INFO } "
2018-07-06 04:48:36 +00:00
# Otherwise, if the GUA_ADDRESS has a value,
elif [ [ ! -z " ${ GUA_ADDRESS } " ] ] ; then
# Let the user know
2021-10-02 21:29:23 +00:00
printf " %b Found IPv6 GUA address\\n" " ${ INFO } "
2018-07-06 04:48:36 +00:00
# And assign it to the global variable
IPV6_ADDRESS = " ${ GUA_ADDRESS } "
# If none of those work,
else
2021-10-02 21:29:23 +00:00
printf " %b Unable to find IPv6 ULA/GUA address\\n" " ${ INFO } "
2018-07-06 04:48:36 +00:00
# So set the variable to be empty
IPV6_ADDRESS = ""
fi
2015-11-10 00:33:32 +00:00
}
2021-10-02 21:29:23 +00:00
# A function to collect IPv4 and IPv6 information of the device
collect_v4andv6_information( ) {
find_IPv4_information
# Echo the information to the user
printf " %b IPv4 address: %s\\n" " ${ INFO } " " ${ IPV4_ADDRESS } "
# if `dhcpcd` is used offer to set this as static IP for the device
if [ [ -f "/etc/dhcpcd.conf" ] ] ; then
2021-11-25 06:41:40 +00:00
# configure networking via dhcpcd
getStaticIPv4Settings
2018-07-06 04:48:36 +00:00
fi
2021-10-02 21:29:23 +00:00
find_IPv6_information
2018-10-31 18:14:05 +00:00
printf " %b IPv6 address: %s\\n" " ${ INFO } " " ${ IPV6_ADDRESS } "
2015-11-10 00:33:32 +00:00
}
2015-11-08 23:21:02 +00:00
2016-01-27 06:11:38 +00:00
getStaticIPv4Settings( ) {
2018-07-06 04:48:36 +00:00
# Local, named variables
local ipSettingsCorrect
2021-11-06 20:32:03 +00:00
local DHCPChoice
2018-07-06 04:48:36 +00:00
# Ask if the user wants to use DHCP settings as their static IP
2022-06-19 00:17:10 +00:00
# This is useful for users that are using DHCP reservations; we can use the information gathered
2022-07-04 21:00:16 +00:00
DHCPChoice = $( dialog --no-shadow --keep-tite --output-fd 1 \
2022-07-05 21:23:59 +00:00
--cancel-label "Exit" --ok-label "Continue" \
2022-06-19 00:17:10 +00:00
--backtitle "Calibrating network interface" \
--title "Static IP Address" \
--menu " Do you want to use your current network settings as a static address?\\n \
IP address: ${ IPV4_ADDRESS } \\ n \
Gateway: ${ IPv4gw } \\ n" \
" ${ r } " " ${ c } " 3 \
"Yes" "Set static IP using current values" \
"No" "Set static IP using custom values" \
"Skip" "I will set a static IP later, or have already done so" )
result = $?
case ${ result } in
" ${ DIALOG_CANCEL } " | " ${ DIALOG_ESC } " )
2022-07-06 10:29:03 +00:00
printf " %b Cancel was selected, exiting installer%b\\n" " ${ COL_LIGHT_RED } " " ${ COL_NC } "
2022-06-19 00:17:10 +00:00
exit 1
; ;
esac
case ${ DHCPChoice } in
"Skip" )
return
; ;
"Yes" )
# If they choose yes, let the user know that the IP address will not be available via DHCP and may cause a conflict.
2022-07-04 21:00:16 +00:00
dialog --no-shadow --keep-tite \
2022-07-05 21:23:59 +00:00
--cancel-label "Exit" \
2022-06-19 00:17:10 +00:00
--backtitle "IP information" \
--title "FYI: IP Conflict" \
2022-07-02 19:36:16 +00:00
--msgbox " \\nIt is possible your router could still try to assign this IP to a device, which would cause a conflict\
But in most cases the router is smart enough to not do that.\
If you are worried, either manually set the address, or modify the DHCP reservation pool so it does not include the IP you want.\
It is also possible to use a DHCP reservation, but if you are going to do that, you might as well set a static address." \
2022-07-04 21:00:16 +00:00
" ${ r } " " ${ c } " && result = 0 || result = $?
2022-06-19 00:17:10 +00:00
case ${ result } in
" ${ DIALOG_CANCEL } " | " ${ DIALOG_ESC } " )
2022-07-06 10:29:03 +00:00
printf " %b Cancel was selected, exiting installer%b\\n" " ${ COL_LIGHT_RED } " " ${ COL_NC } "
2022-06-19 00:17:10 +00:00
exit 1
; ;
esac
; ;
"No" )
# Otherwise, we need to ask the user to input their desired settings.
# Start by getting the IPv4 address (pre-filling it with info gathered from DHCP)
# Start a loop to let the user enter their information with the chance to go back and edit it if necessary
ipSettingsCorrect = false
until [ [ " ${ ipSettingsCorrect } " = True ] ] ; do
# Ask for the IPv4 address
2022-07-04 21:00:16 +00:00
_staticIPv4Temp = $( dialog --no-shadow --keep-tite --output-fd 1 \
2022-07-05 21:23:59 +00:00
--cancer-label "Exit" \
--ok-label "Continue" \
2022-06-19 00:17:10 +00:00
--backtitle "Calibrating network interface" \
--title "IPv4 Address" \
--form "\\nEnter your desired IPv4 address" \
" ${ r } " " ${ c } " 0 \
"IPv4 Address:" 1 1 " ${ IPV4_ADDRESS } " 1 15 19 0 \
"IPv4 Gateway:" 2 1 " ${ IPv4gw } " 2 15 19 0)
result = $?
case ${ result } in
" ${ DIALOG_CANCEL } " | " ${ DIALOG_ESC } " )
2022-07-06 10:29:03 +00:00
printf " %b Cancel was selected, exiting installer%b\\n" " ${ COL_LIGHT_RED } " " ${ COL_NC } "
2022-06-19 00:17:10 +00:00
exit 1
; ;
esac
IPV4_ADDRESS = ${ _staticIPv4Temp % $'\n' * }
IPv4gw = ${ _staticIPv4Temp #* $'\n' }
# Give the user a chance to review their settings before moving on
2022-07-04 21:00:16 +00:00
dialog --no-shadow --keep-tite \
2022-07-05 21:23:59 +00:00
--no-label "Edit IP" \
2022-06-19 00:17:10 +00:00
--backtitle "Calibrating network interface" \
--title "Static IP Address" \
--defaultno \
--yesno " Are these settings correct?
IP address: ${ IPV4_ADDRESS }
Gateway: ${ IPv4gw } " \
2022-07-05 21:23:59 +00:00
" ${ r } " " ${ c } " && ipSettingsCorrect = True
2021-11-06 20:32:03 +00:00
done
2022-06-19 00:17:10 +00:00
; ;
esac
setDHCPCD
2015-11-08 23:21:02 +00:00
}
2021-03-18 01:35:27 +00:00
# Configure networking via dhcpcd
2016-01-27 06:11:38 +00:00
setDHCPCD( ) {
2021-03-18 01:35:27 +00:00
# Check if the IP is already in the file
2018-11-09 19:06:47 +00:00
if grep -q " ${ IPV4_ADDRESS } " /etc/dhcpcd.conf; then
printf " %b Static IP already configured\\n" " ${ INFO } "
# If it's not,
else
# we can append these lines to dhcpcd.conf to enable a static IP
echo " interface ${ PIHOLE_INTERFACE }
static ip_address = ${ IPV4_ADDRESS }
static routers = ${ IPv4gw }
2020-03-12 18:48:40 +00:00
static domain_name_servers = ${ PIHOLE_DNS_1 } ${ PIHOLE_DNS_2 } " | tee -a /etc/dhcpcd.conf >/dev/null
2018-11-09 19:06:47 +00:00
# Then use the ip command to immediately set the new address
ip addr replace dev " ${ PIHOLE_INTERFACE } " " ${ IPV4_ADDRESS } "
# Also give a warning that the user may need to reboot their system
2019-11-07 12:59:44 +00:00
printf " %b Set IP address to %s\\n" " ${ TICK } " " ${ IPV4_ADDRESS %/* } "
printf " %b You may need to restart after the install is complete\\n" " ${ INFO } "
2018-11-09 19:06:47 +00:00
fi
2015-10-31 14:11:29 +00:00
}
2017-07-07 00:25:56 +00:00
# Check an IP address to see if it is a valid one
2016-10-20 02:47:45 +00:00
valid_ip( ) {
2018-07-06 04:48:36 +00:00
# Local, named variables
local ip = ${ 1 }
local stat = 1
2021-03-18 01:35:27 +00:00
# Regex matching one IPv4 component, i.e. an integer from 0 to 255.
2021-03-18 01:09:03 +00:00
# See https://tools.ietf.org/html/rfc1340
2020-06-15 21:52:24 +00:00
local ipv4elem = "(25[0-5]|2[0-4][0-9]|1[0-9][0-9]|[1-9][0-9]?|0)" ;
2021-04-14 20:50:04 +00:00
# Regex matching an optional port (starting with '#') range of 1-65536
local portelem = "(#(6553[0-5]|655[0-2][0-9]|65[0-4][0-9]{2}|6[0-4][0-9]{3}|[1-5][0-9]{4}|[1-9][0-9]{0,3}|0))?" ;
2021-03-18 01:35:27 +00:00
# Build a full IPv4 regex from the above subexpressions
2021-12-02 13:44:50 +00:00
local regex = " ^ ${ ipv4elem } \\. ${ ipv4elem } \\. ${ ipv4elem } \\. ${ ipv4elem } ${ portelem } $"
2020-06-13 19:30:03 +00:00
2021-03-18 01:35:27 +00:00
# Evaluate the regex, and return the result
2020-06-15 21:52:24 +00:00
[ [ $ip = ~ ${ regex } ] ]
stat = $?
2019-05-10 14:13:23 +00:00
return " ${ stat } "
2016-02-20 17:33:20 +00:00
}
2020-06-23 19:38:18 +00:00
valid_ip6( ) {
local ip = ${ 1 }
local stat = 1
2021-03-18 01:35:27 +00:00
# Regex matching one IPv6 element, i.e. a hex value from 0000 to FFFF
2020-06-23 19:38:18 +00:00
local ipv6elem = "[0-9a-fA-F]{1,4}"
2021-03-18 01:35:27 +00:00
# Regex matching an IPv6 CIDR, i.e. 1 to 128
2020-06-23 19:38:18 +00:00
local v6cidr = "(\\/([1-9]|[1-9][0-9]|1[0-1][0-9]|12[0-8])){0,1}"
2021-04-14 20:50:04 +00:00
# Regex matching an optional port (starting with '#') range of 1-65536
local portelem = "(#(6553[0-5]|655[0-2][0-9]|65[0-4][0-9]{2}|6[0-4][0-9]{3}|[1-5][0-9]{4}|[1-9][0-9]{0,3}|0))?" ;
2021-03-18 01:35:27 +00:00
# Build a full IPv6 regex from the above subexpressions
2020-08-04 20:11:32 +00:00
local regex = " ^((( ${ ipv6elem } ))*((: ${ ipv6elem } ))*::(( ${ ipv6elem } ))*((: ${ ipv6elem } ))*|(( ${ ipv6elem } ))((: ${ ipv6elem } )){7}) ${ v6cidr } ${ portelem } $"
2020-06-23 19:38:18 +00:00
2021-03-18 01:35:27 +00:00
# Evaluate the regex, and return the result
2020-06-23 19:38:18 +00:00
[ [ ${ ip } = ~ ${ regex } ] ]
stat = $?
return " ${ stat } "
}
2017-07-07 00:25:56 +00:00
# A function to choose the upstream DNS provider(s)
2016-10-20 02:47:45 +00:00
setDNS( ) {
2018-07-06 04:48:36 +00:00
# Local, named variables
local DNSSettingsCorrect
# In an array, list the available upstream providers
2019-01-13 14:51:30 +00:00
DNSChooseOptions = ( )
local DNSServerCount = 0
2021-03-18 01:35:27 +00:00
# Save the old Internal Field Separator in a variable,
2019-01-13 14:51:30 +00:00
OIFS = $IFS
# and set the new one to newline
IFS = $'\n'
# Put the DNS Servers into an array
for DNSServer in ${ DNS_SERVERS }
do
DNSName = " $( cut -d';' -f1 <<< " ${ DNSServer } " ) "
DNSChooseOptions[ DNSServerCount] = " ${ DNSName } "
( ( DNSServerCount = DNSServerCount+1 ) )
DNSChooseOptions[ DNSServerCount] = ""
( ( DNSServerCount = DNSServerCount+1 ) )
done
DNSChooseOptions[ DNSServerCount] = "Custom"
( ( DNSServerCount = DNSServerCount+1 ) )
DNSChooseOptions[ DNSServerCount] = ""
# Restore the IFS to what it was
IFS = ${ OIFS }
2022-06-19 00:17:10 +00:00
# In a dialog, show the options
2022-07-04 21:00:16 +00:00
DNSchoices = $( dialog --no-shadow --keep-tite --output-fd 1 \
2022-07-05 21:23:59 +00:00
--cancel-label "Exit" \
2022-06-19 00:17:10 +00:00
--menu "Select Upstream DNS Provider. To use your own, select Custom." " ${ r } " " ${ c } " 7 \
" ${ DNSChooseOptions [@] } " )
result = $?
case ${ result } in
" ${ DIALOG_CANCEL } " | " ${ DIALOG_ESC } " )
2022-07-06 10:29:03 +00:00
printf " %b Cancel was selected, exiting installer%b\\n" " ${ COL_LIGHT_RED } " " ${ COL_NC } "
2022-06-19 00:17:10 +00:00
exit 1
; ;
esac
2017-06-21 11:49:05 +00:00
2021-03-18 01:35:27 +00:00
# Depending on the user's choice, set the GLOBAL variables to the IP of the respective provider
2019-01-13 14:51:30 +00:00
if [ [ " ${ DNSchoices } " = = "Custom" ] ]
then
2021-03-18 01:35:27 +00:00
# Loop until we have a valid DNS setting
2019-01-13 14:51:30 +00:00
until [ [ " ${ DNSSettingsCorrect } " = True ] ] ; do
2021-03-18 01:35:27 +00:00
# Signal value, to be used if the user inputs an invalid IP address
2019-01-13 14:51:30 +00:00
strInvalid = "Invalid"
if [ [ ! " ${ PIHOLE_DNS_1 } " ] ] ; then
if [ [ ! " ${ PIHOLE_DNS_2 } " ] ] ; then
2021-03-18 01:35:27 +00:00
# If the first and second upstream servers do not exist, do not prepopulate an IP address
2019-01-13 14:51:30 +00:00
prePopulate = ""
else
2022-06-19 00:17:10 +00:00
# Otherwise, prepopulate the dialogue with the appropriate DNS value(s)
2019-01-13 14:51:30 +00:00
prePopulate = " , ${ PIHOLE_DNS_2 } "
2018-07-06 04:48:36 +00:00
fi
2019-01-13 14:51:30 +00:00
elif [ [ " ${ PIHOLE_DNS_1 } " ] ] && [ [ ! " ${ PIHOLE_DNS_2 } " ] ] ; then
prePopulate = " ${ PIHOLE_DNS_1 } "
elif [ [ " ${ PIHOLE_DNS_1 } " ] ] && [ [ " ${ PIHOLE_DNS_2 } " ] ] ; then
prePopulate = " ${ PIHOLE_DNS_1 } , ${ PIHOLE_DNS_2 } "
fi
2018-07-06 04:48:36 +00:00
2021-03-18 01:35:27 +00:00
# Prompt the user to enter custom upstream servers
2022-07-04 21:00:16 +00:00
piholeDNS = $( dialog --no-shadow --keep-tite --output-fd 1 \
2022-07-05 21:23:59 +00:00
--cancel-label "Exit" \
2022-06-19 00:17:10 +00:00
--backtitle "Specify Upstream DNS Provider(s)" \
2022-07-02 19:36:16 +00:00
--inputbox " Enter your desired upstream DNS provider(s), separated by a comma.\
If you want to specify a port other than 53, separate it with a hash.\
\\ n\\ nFor example '8.8.8.8, 8.8.4.4' or '127.0.0.1#5335' " \
2022-06-19 00:17:10 +00:00
" ${ r } " " ${ c } " " ${ prePopulate } " )
result = $?
case ${ result } in
" ${ DIALOG_CANCEL } " | " ${ DIALOG_ESC } " )
2022-07-06 10:29:03 +00:00
printf " %b Cancel was selected, exiting installer%b\\n" " ${ COL_LIGHT_RED } " " ${ COL_NC } "
2022-06-19 00:17:10 +00:00
exit 1
; ;
esac
2019-01-13 14:51:30 +00:00
# Clean user input and replace whitespace with comma.
piholeDNS = $( sed 's/[, \t]\+/,/g' <<< " ${ piholeDNS } " )
2018-10-31 18:14:05 +00:00
2021-03-18 01:35:27 +00:00
# Separate the user input into the two DNS values (separated by a comma)
2019-01-13 14:51:30 +00:00
printf -v PIHOLE_DNS_1 "%s" " ${ piholeDNS %%,* } "
printf -v PIHOLE_DNS_2 "%s" " ${ piholeDNS ##*, } "
2018-10-31 18:14:05 +00:00
2021-03-18 01:35:27 +00:00
# If the first DNS value is invalid or empty, this if statement will be true and we will set PIHOLE_DNS_1="Invalid"
2019-01-13 14:51:30 +00:00
if ! valid_ip " ${ PIHOLE_DNS_1 } " || [ [ ! " ${ PIHOLE_DNS_1 } " ] ] ; then
PIHOLE_DNS_1 = ${ strInvalid }
fi
2021-03-18 01:35:27 +00:00
# If the second DNS value is invalid or empty, this if statement will be true and we will set PIHOLE_DNS_2="Invalid"
2019-01-13 14:51:30 +00:00
if ! valid_ip " ${ PIHOLE_DNS_2 } " && [ [ " ${ PIHOLE_DNS_2 } " ] ] ; then
PIHOLE_DNS_2 = ${ strInvalid }
fi
# If either of the DNS servers are invalid,
if [ [ " ${ PIHOLE_DNS_1 } " = = " ${ strInvalid } " ] ] || [ [ " ${ PIHOLE_DNS_2 } " = = " ${ strInvalid } " ] ] ; then
2021-03-18 01:35:27 +00:00
# explain this to the user,
2022-07-04 21:00:16 +00:00
dialog --no-shadow --keep-tite \
2022-06-19 00:17:10 +00:00
--title "Invalid IP Address(es)" \
--backtitle "Invalid IP" \
2022-07-02 19:36:16 +00:00
--msgbox " \\nOne or both of the entered IP addresses were invalid. Please try again.\
\\ n\\ nInvalid IPs: ${ PIHOLE_DNS_1 } , ${ PIHOLE_DNS_2 } " \
2022-07-05 21:23:59 +00:00
" ${ r } " " ${ c } "
2022-06-19 00:17:10 +00:00
2021-03-18 01:35:27 +00:00
# set the variables back to nothing,
2019-01-13 14:51:30 +00:00
if [ [ " ${ PIHOLE_DNS_1 } " = = " ${ strInvalid } " ] ] ; then
PIHOLE_DNS_1 = ""
2018-07-06 04:48:36 +00:00
fi
2019-01-13 14:51:30 +00:00
if [ [ " ${ PIHOLE_DNS_2 } " = = " ${ strInvalid } " ] ] ; then
PIHOLE_DNS_2 = ""
2018-07-06 04:48:36 +00:00
fi
2021-03-18 01:35:27 +00:00
# and continue the loop.
2019-11-07 12:59:44 +00:00
DNSSettingsCorrect = False
2019-01-13 14:51:30 +00:00
else
2022-07-04 21:00:16 +00:00
dialog --no-shadow --keep-tite \
2022-06-19 00:17:10 +00:00
--backtitle "Specify Upstream DNS Provider(s)" \
--title "Upstream DNS Provider(s)" \
--yesno " Are these settings correct?\\n\\tDNS Server 1:\\t ${ PIHOLE_DNS_1 } \\n\\tDNS Server 2:\\t ${ PIHOLE_DNS_2 } " \
2022-07-04 21:00:16 +00:00
" ${ r } " " ${ c } " && result = 0 || result = $?
2022-06-19 00:17:10 +00:00
case ${ result } in
" ${ DIALOG_OK } " )
DNSSettingsCorrect = True
; ;
" ${ DIALOG_CANCEL } " )
DNSSettingsCorrect = False
; ;
" ${ DIALOG_ESC } " )
2022-07-06 03:25:27 +00:00
printf " %b Escape pressed, exiting installer at DNS Settings%b\\n" " ${ COL_LIGHT_RED } " " ${ COL_NC } "
2022-06-19 00:17:10 +00:00
exit 1
; ;
esac
2019-01-13 14:51:30 +00:00
fi
done
2019-11-07 12:59:44 +00:00
else
2021-03-18 01:35:27 +00:00
# Save the old Internal Field Separator in a variable,
2019-01-13 14:51:30 +00:00
OIFS = $IFS
# and set the new one to newline
IFS = $'\n'
for DNSServer in ${ DNS_SERVERS }
do
DNSName = " $( cut -d';' -f1 <<< " ${ DNSServer } " ) "
if [ [ " ${ DNSchoices } " = = " ${ DNSName } " ] ]
then
PIHOLE_DNS_1 = " $( cut -d';' -f2 <<< " ${ DNSServer } " ) "
PIHOLE_DNS_2 = " $( cut -d';' -f3 <<< " ${ DNSServer } " ) "
break
fi
done
# Restore the IFS to what it was
IFS = ${ OIFS }
fi
2019-11-07 12:59:44 +00:00
# Display final selection
2019-12-04 17:59:25 +00:00
local DNSIP = ${ PIHOLE_DNS_1 }
[ [ -z ${ PIHOLE_DNS_2 } ] ] || DNSIP += " , ${ PIHOLE_DNS_2 } "
printf " %b Using upstream DNS: %s (%s)\\n" " ${ INFO } " " ${ DNSchoices } " " ${ DNSIP } "
2016-01-25 05:11:00 +00:00
}
2017-07-07 00:25:56 +00:00
# Allow the user to enable/disable logging
2016-10-31 21:38:48 +00:00
setLogging( ) {
2022-06-19 00:17:10 +00:00
# Ask the user if they want to enable logging
2022-07-04 21:00:16 +00:00
dialog --no-shadow --keep-tite \
2022-06-19 00:17:10 +00:00
--backtitle "Pihole Installation" \
--title "Enable Logging" \
--yesno "\\n\\nWould you like to enable query logging?" \
2022-07-04 21:00:16 +00:00
" ${ r } " " ${ c } " && result = 0 || result = $?
2022-06-19 00:17:10 +00:00
case ${ result } in
" ${ DIALOG_OK } " )
# If they chose yes,
2022-07-06 03:11:00 +00:00
printf " %b Query Logging on.\\n" " ${ INFO } "
2018-07-06 04:48:36 +00:00
QUERY_LOGGING = true
; ;
2022-06-19 00:17:10 +00:00
" ${ DIALOG_CANCEL } " )
# If they chose no,
2022-07-06 03:11:00 +00:00
printf " %b Query Logging off.\\n" " ${ INFO } "
2018-07-06 04:48:36 +00:00
QUERY_LOGGING = false
; ;
2022-06-19 00:17:10 +00:00
" ${ DIALOG_ESC } " )
# User pressed <ESC>
2022-07-06 03:25:27 +00:00
printf " %b Escape pressed, exiting installer at Query Logging choice.%b\\n" " ${ COL_LIGHT_RED } " " ${ COL_NC } "
2022-06-19 00:17:10 +00:00
exit 1
; ;
2016-12-21 01:22:57 +00:00
esac
2016-10-31 21:38:48 +00:00
}
2018-08-20 23:04:58 +00:00
# Allow the user to set their FTL privacy level
setPrivacyLevel( ) {
# The default selection is level 0
2022-07-04 21:00:16 +00:00
PRIVACY_LEVEL = $( dialog --no-shadow --keep-tite --output-fd 1 \
2022-07-05 21:23:59 +00:00
--cancel-label "Exit" \
--ok-label "Continue" \
2022-06-19 00:17:10 +00:00
--radiolist "Select a privacy mode for FTL. https://docs.pi-hole.net/ftldns/privacylevels/" \
" ${ r } " " ${ c } " 6 \
"0" "Show everything" on \
"1" "Hide domains" off \
"2" "Hide domains and clients" off \
"3" "Anonymous mode" off)
result = $?
case ${ result } in
" ${ DIALOG_OK } " )
printf " %b Using privacy level: %s\\n" " ${ INFO } " " ${ PRIVACY_LEVEL } "
; ;
" ${ DIALOG_CANCEL } " | " ${ DIALOG_ESC } " )
printf " %b Cancelled privacy level selection.%b\\n" " ${ COL_LIGHT_RED } " " ${ COL_NC } "
exit 1
; ;
esac
2018-08-20 23:04:58 +00:00
}
2018-02-07 05:45:23 +00:00
# Function to ask the user if they want to install the dashboard
2017-01-28 14:38:54 +00:00
setAdminFlag( ) {
2018-07-06 04:48:36 +00:00
# Similar to the logging function, ask what the user wants
2022-07-04 21:00:16 +00:00
dialog --no-shadow --keep-tite \
2022-06-19 00:17:10 +00:00
--backtitle "Pihole Installation" \
--title "Admin Web Interface" \
--yesno "\\n\\nDo you want to install the Admin Web Interface?" \
2022-07-04 21:00:16 +00:00
" ${ r } " " ${ c } " && result = 0 || result = $?
2022-06-19 00:17:10 +00:00
case ${ result } in
" ${ DIALOG_OK } " )
# If they chose yes,
2022-07-06 03:11:00 +00:00
printf " %b Installing Admin Web Interface\\n" " ${ INFO } "
2022-06-19 00:17:10 +00:00
# Set the flag to install the web interface
2018-07-06 04:48:36 +00:00
INSTALL_WEB_INTERFACE = true
; ;
2022-06-19 00:17:10 +00:00
" ${ DIALOG_CANCEL } " )
# If they chose no,
2022-07-06 03:11:00 +00:00
printf " %b Not installing Admin Web Interface\\n" " ${ INFO } "
2022-06-19 00:17:10 +00:00
# Set the flag to not install the web interface
2018-07-06 04:48:36 +00:00
INSTALL_WEB_INTERFACE = false
2020-10-09 14:12:21 +00:00
INSTALL_WEB_SERVER = false
2018-07-06 04:48:36 +00:00
; ;
2022-06-19 00:17:10 +00:00
" ${ DIALOG_ESC } " )
# User pressed <ESC>
2022-07-06 03:25:27 +00:00
printf " %b Escape pressed, exiting installer at Admin Web Interface choice.%b\\n" " ${ COL_LIGHT_RED } " " ${ COL_NC } "
2022-06-19 00:17:10 +00:00
exit 1
; ;
2017-01-28 14:38:54 +00:00
esac
2018-07-06 04:48:36 +00:00
2021-03-18 01:35:27 +00:00
# If the user wants to install the Web admin interface (i.e. it has not been deselected above)
2022-06-19 00:17:10 +00:00
if [ [ " ${ INSTALL_WEB_INTERFACE } " = = true ] ] ; then
2020-10-09 14:12:21 +00:00
# Get list of required PHP modules, excluding base package (common) and handler (cgi)
local i php_modules
for i in " ${ PIHOLE_WEB_DEPS [@] } " ; do [ [ $i = = 'php' * && $i != *'-common' && $i != *'-cgi' ] ] && php_modules += " ${ i #*- } " ; done
2022-07-04 21:00:16 +00:00
dialog --no-shadow --keep-tite \
2022-06-19 00:17:10 +00:00
--backtitle "Pi-hole Installation" \
--title "Web Server" \
2022-07-02 19:36:16 +00:00
--yesno " \\n\\nA web server is required for the Admin Web Interface.\
\\ n\\ nDo you want to install lighttpd and the required PHP modules?\
2022-07-05 07:06:12 +00:00
\\ n\\ nNB: If you disable this, and, do not have an existing web server \
and required PHP modules ( ${ php_modules # } ) installed, the web interface \
will not function . Additionally the web server user needs to be member of \
2022-07-02 19:36:16 +00:00
the \" pihole\" group for full functionality." \
2022-07-04 21:00:16 +00:00
" ${ r } " " ${ c } " && result = 0 || result = $?
2022-06-19 00:17:10 +00:00
case ${ result } in
" ${ DIALOG_OK } " )
# If they chose yes,
2022-07-06 03:11:00 +00:00
printf " %b Installing lighttpd\\n" " ${ INFO } "
2022-06-19 00:17:10 +00:00
# Set the flag to install the web server
2018-07-06 04:48:36 +00:00
INSTALL_WEB_SERVER = true
; ;
2022-06-19 00:17:10 +00:00
" ${ DIALOG_CANCEL } " )
# If they chose no,
2022-07-06 03:11:00 +00:00
printf " %b Not installing lighttpd\\n" " ${ INFO } "
2022-06-19 00:17:10 +00:00
# Set the flag to not install the web server
2018-07-06 04:48:36 +00:00
INSTALL_WEB_SERVER = false
; ;
2022-06-19 00:17:10 +00:00
" ${ DIALOG_ESC } " )
# User pressed <ESC>
2022-07-06 03:25:27 +00:00
printf " %b Escape pressed, exiting installer at web server choice.%b\\n" " ${ COL_LIGHT_RED } " " ${ COL_NC } "
2022-06-19 00:17:10 +00:00
exit 1
; ;
2018-07-06 04:48:36 +00:00
esac
fi
2017-01-28 14:38:54 +00:00
}
2018-04-17 08:50:25 +00:00
# A function to display a list of example blocklists for users to select
2018-04-17 18:35:46 +00:00
chooseBlocklists( ) {
2018-07-06 04:48:36 +00:00
# Back up any existing adlist file, on the off chance that it exists. Useful in case of a reconfigure.
if [ [ -f " ${ adlistFile } " ] ] ; then
mv " ${ adlistFile } " " ${ adlistFile } .old "
fi
2022-06-19 00:17:10 +00:00
# Let user select (or not) blocklists
2022-07-04 21:00:16 +00:00
dialog --no-shadow --keep-tite \
2022-06-19 00:17:10 +00:00
--backtitle "Pi-hole Installation" \
--title "Blocklists" \
2022-07-02 19:36:16 +00:00
--yesno " \\nPi-hole relies on third party lists in order to block ads.\
\\ n\\ nYou can use the suggestion below, and/or add your own after installation.\
\\ n\\ nSelect 'Yes' to include:\
\\ n\\ nStevenBlack' s Unified Hosts List" \
2022-07-04 21:00:16 +00:00
" ${ r } " " ${ c } " && result = 0 || result = $?
2022-06-19 00:17:10 +00:00
case ${ result } in
" ${ DIALOG_OK } " )
# If they chose yes,
2022-07-06 03:11:00 +00:00
printf " %b Installing StevenBlack's Unified Hosts List\\n" " ${ INFO } "
2022-06-19 00:17:10 +00:00
echo "https://raw.githubusercontent.com/StevenBlack/hosts/master/hosts" >> " ${ adlistFile } "
; ;
" ${ DIALOG_CANCEL } " )
# If they chose no,
2022-07-06 03:11:00 +00:00
printf " %b Not installing StevenBlack's Unified Hosts List\\n" " ${ INFO } "
2022-06-19 00:17:10 +00:00
; ;
" ${ DIALOG_ESC } " )
# User pressed <ESC>
2022-07-06 03:25:27 +00:00
printf " %b Escape pressed, exiting installer at blocklist choice.%b\\n" " ${ COL_LIGHT_RED } " " ${ COL_NC } "
2022-06-19 00:17:10 +00:00
exit 1
; ;
esac
2021-03-18 01:35:27 +00:00
# Create an empty adList file with appropriate permissions.
2022-03-31 19:03:17 +00:00
if [ ! -f " ${ adlistFile } " ] ; then
2022-03-31 21:32:07 +00:00
install -m 644 /dev/null " ${ adlistFile } "
else
2022-03-31 19:03:17 +00:00
chmod 644 " ${ adlistFile } "
fi
2017-01-28 14:38:54 +00:00
}
2018-08-12 21:19:16 +00:00
# Used only in unattended setup
# If there is already the adListFile, we keep it, else we create it using all default lists
installDefaultBlocklists( ) {
2018-08-14 12:59:41 +00:00
# In unattended setup, could be useful to use userdefined blocklist.
# If this file exists, we avoid overriding it.
2018-08-12 21:19:16 +00:00
if [ [ -f " ${ adlistFile } " ] ] ; then
return ;
2018-08-20 20:52:22 +00:00
fi
2022-06-19 00:17:10 +00:00
echo "https://raw.githubusercontent.com/StevenBlack/hosts/master/hosts" >> " ${ adlistFile } "
2018-08-12 21:19:16 +00:00
}
2018-02-07 05:45:23 +00:00
# Check if /etc/dnsmasq.conf is from pi-hole. If so replace with an original and install new in .d directory
2016-10-20 02:47:45 +00:00
version_check_dnsmasq( ) {
2018-07-06 04:48:36 +00:00
# Local, named variables
local dnsmasq_conf = "/etc/dnsmasq.conf"
local dnsmasq_conf_orig = "/etc/dnsmasq.conf.orig"
local dnsmasq_pihole_id_string = "addn-hosts=/etc/pihole/gravity.list"
2019-04-26 14:16:30 +00:00
local dnsmasq_pihole_id_string2 = "# Dnsmasq config for Pi-hole's FTLDNS"
2018-07-06 04:48:36 +00:00
local dnsmasq_original_config = " ${ PI_HOLE_LOCAL_REPO } /advanced/dnsmasq.conf.original "
2021-08-16 07:57:12 +00:00
local dnsmasq_pihole_01_source = " ${ PI_HOLE_LOCAL_REPO } /advanced/01-pihole.conf "
local dnsmasq_pihole_01_target = "/etc/dnsmasq.d/01-pihole.conf"
local dnsmasq_rfc6761_06_source = " ${ PI_HOLE_LOCAL_REPO } /advanced/06-rfc6761.conf "
local dnsmasq_rfc6761_06_target = "/etc/dnsmasq.d/06-rfc6761.conf"
2018-07-06 04:48:36 +00:00
# If the dnsmasq config file exists
if [ [ -f " ${ dnsmasq_conf } " ] ] ; then
2018-10-31 18:14:05 +00:00
printf " %b Existing dnsmasq.conf found..." " ${ INFO } "
2019-04-24 16:54:55 +00:00
# If a specific string is found within this file, we presume it's from older versions on Pi-hole,
2019-04-26 14:16:30 +00:00
if grep -q " ${ dnsmasq_pihole_id_string } " " ${ dnsmasq_conf } " ||
grep -q " ${ dnsmasq_pihole_id_string2 } " " ${ dnsmasq_conf } " ; then
2018-10-31 18:14:05 +00:00
printf " it is from a previous Pi-hole install.\\n"
printf " %b Backing up dnsmasq.conf to dnsmasq.conf.orig..." " ${ INFO } "
2021-03-18 01:35:27 +00:00
# so backup the original file,
2019-05-10 14:13:23 +00:00
mv -f " ${ dnsmasq_conf } " " ${ dnsmasq_conf_orig } "
2018-10-31 18:14:05 +00:00
printf "%b %b Backing up dnsmasq.conf to dnsmasq.conf.orig...\\n" " ${ OVER } " " ${ TICK } "
printf " %b Restoring default dnsmasq.conf..." " ${ INFO } "
2018-07-06 04:48:36 +00:00
# and replace it with the default
2019-05-10 14:13:23 +00:00
install -D -m 644 -T " ${ dnsmasq_original_config } " " ${ dnsmasq_conf } "
2018-10-31 18:14:05 +00:00
printf "%b %b Restoring default dnsmasq.conf...\\n" " ${ OVER } " " ${ TICK } "
2018-07-06 04:48:36 +00:00
else
2021-11-25 06:41:40 +00:00
# Otherwise, don't to anything
printf " it is not a Pi-hole file, leaving alone!\\n"
2018-07-06 04:48:36 +00:00
fi
2016-12-21 01:22:57 +00:00
else
2018-07-06 04:48:36 +00:00
# If a file cannot be found,
2018-10-31 18:14:05 +00:00
printf " %b No dnsmasq.conf found... restoring default dnsmasq.conf..." " ${ INFO } "
2018-07-06 04:48:36 +00:00
# restore the default one
2019-05-10 14:13:23 +00:00
install -D -m 644 -T " ${ dnsmasq_original_config } " " ${ dnsmasq_conf } "
2018-10-31 18:14:05 +00:00
printf "%b %b No dnsmasq.conf found... restoring default dnsmasq.conf...\\n" " ${ OVER } " " ${ TICK } "
2018-07-06 04:48:36 +00:00
fi
2016-12-21 01:22:57 +00:00
2021-08-16 07:57:12 +00:00
printf " %b Installing %s..." " ${ INFO } " " ${ dnsmasq_pihole_01_target } "
2018-07-06 04:48:36 +00:00
# Check to see if dnsmasq directory exists (it may not due to being a fresh install and dnsmasq no longer being a dependency)
if [ [ ! -d "/etc/dnsmasq.d" ] ] ; then
2019-05-03 10:27:56 +00:00
install -d -m 755 "/etc/dnsmasq.d"
2018-07-06 04:48:36 +00:00
fi
# Copy the new Pi-hole DNS config file into the dnsmasq.d directory
2021-08-16 07:57:12 +00:00
install -D -m 644 -T " ${ dnsmasq_pihole_01_source } " " ${ dnsmasq_pihole_01_target } "
printf "%b %b Installed %s\n" " ${ OVER } " " ${ TICK } " " ${ dnsmasq_pihole_01_target } "
2018-07-06 04:48:36 +00:00
# Replace our placeholder values with the GLOBAL DNS variables that we populated earlier
2021-03-18 01:35:27 +00:00
# First, swap in the interface to listen on,
2021-08-16 07:57:12 +00:00
sed -i " s/@INT@/ $PIHOLE_INTERFACE / " " ${ dnsmasq_pihole_01_target } "
2018-07-06 04:48:36 +00:00
if [ [ " ${ PIHOLE_DNS_1 } " != "" ] ] ; then
2021-03-18 01:35:27 +00:00
# then swap in the primary DNS server.
2021-08-16 07:57:12 +00:00
sed -i " s/@DNS1@/ $PIHOLE_DNS_1 / " " ${ dnsmasq_pihole_01_target } "
2018-07-06 04:48:36 +00:00
else
2021-03-18 01:35:27 +00:00
# Otherwise, remove the line which sets DNS1.
2021-08-16 07:57:12 +00:00
sed -i '/^server=@DNS1@/d' " ${ dnsmasq_pihole_01_target } "
2018-07-06 04:48:36 +00:00
fi
2021-03-18 01:35:27 +00:00
# Ditto if DNS2 is not empty
2018-07-06 04:48:36 +00:00
if [ [ " ${ PIHOLE_DNS_2 } " != "" ] ] ; then
2021-08-16 07:57:12 +00:00
sed -i " s/@DNS2@/ $PIHOLE_DNS_2 / " " ${ dnsmasq_pihole_01_target } "
2018-07-06 04:48:36 +00:00
else
2021-08-16 07:57:12 +00:00
sed -i '/^server=@DNS2@/d' " ${ dnsmasq_pihole_01_target } "
2018-07-06 04:48:36 +00:00
fi
2016-12-21 01:22:57 +00:00
2021-11-25 06:41:40 +00:00
# Set the cache size
sed -i " s/@CACHE_SIZE@/ $CACHE_SIZE / " " ${ dnsmasq_pihole_01_target } "
2020-09-28 04:15:46 +00:00
2019-05-10 14:13:23 +00:00
sed -i 's/^#conf-dir=\/etc\/dnsmasq.d$/conf-dir=\/etc\/dnsmasq.d/' " ${ dnsmasq_conf } "
2018-07-06 04:48:36 +00:00
# If the user does not want to enable logging,
if [ [ " ${ QUERY_LOGGING } " = = false ] ] ; then
2021-03-18 01:35:27 +00:00
# disable it by commenting out the directive in the DNS config file
2021-08-16 07:57:12 +00:00
sed -i 's/^log-queries/#log-queries/' " ${ dnsmasq_pihole_01_target } "
2016-10-31 21:38:48 +00:00
else
2021-03-18 01:35:27 +00:00
# Otherwise, enable it by uncommenting the directive in the DNS config file
2021-08-16 07:57:12 +00:00
sed -i 's/^#log-queries/log-queries/' " ${ dnsmasq_pihole_01_target } "
2016-10-31 21:38:48 +00:00
fi
2021-08-15 09:17:09 +00:00
2021-08-16 07:57:12 +00:00
printf " %b Installing %s..." " ${ INFO } " " ${ dnsmasq_rfc6761_06_source } "
install -D -m 644 -T " ${ dnsmasq_rfc6761_06_source } " " ${ dnsmasq_rfc6761_06_target } "
printf "%b %b Installed %s\n" " ${ OVER } " " ${ TICK } " " ${ dnsmasq_rfc6761_06_target } "
2016-01-25 05:04:02 +00:00
}
2017-07-10 19:12:30 +00:00
# Clean an existing installation to prepare for upgrade/reinstall
2016-11-02 07:14:25 +00:00
clean_existing( ) {
2018-07-06 04:48:36 +00:00
# Local, named variables
# ${1} Directory to clean
local clean_directory = " ${ 1 } "
2021-03-18 01:35:27 +00:00
# Pop the first argument, and shift all addresses down by one (i.e. ${2} becomes ${1})
2018-07-06 04:48:36 +00:00
shift
2021-03-18 01:35:27 +00:00
# Then, we can access all arguments ($@) without including the directory to clean
2018-07-06 04:48:36 +00:00
local old_files = ( " $@ " )
2021-03-18 01:35:27 +00:00
# Remove each script in the old_files array
2018-07-06 04:48:36 +00:00
for script in " ${ old_files [@] } " ; do
rm -f " ${ clean_directory } / ${ script } .sh "
done
2016-11-02 07:14:25 +00:00
}
2016-10-05 16:57:48 +00:00
2017-07-07 00:25:56 +00:00
# Install the scripts from repository to their various locations
2016-11-02 07:14:25 +00:00
installScripts( ) {
2018-07-06 04:48:36 +00:00
# Local, named variables
local str = " Installing scripts from ${ PI_HOLE_LOCAL_REPO } "
2018-10-31 18:14:05 +00:00
printf " %b %s..." " ${ INFO } " " ${ str } "
2018-07-06 04:48:36 +00:00
# Clear out script files from Pi-hole scripts directory.
clean_existing " ${ PI_HOLE_INSTALL_DIR } " " ${ PI_HOLE_FILES [@] } "
# Install files from local core repository
if is_repo " ${ PI_HOLE_LOCAL_REPO } " ; then
# move into the directory
cd " ${ PI_HOLE_LOCAL_REPO } "
# Install the scripts by:
# -o setting the owner to the user
# -Dm755 create all leading components of destination except the last, then copy the source to the destination and setting the permissions to 755
#
# This first one is the directory
install -o " ${ USER } " -Dm755 -d " ${ PI_HOLE_INSTALL_DIR } "
# The rest are the scripts Pi-hole needs
install -o " ${ USER } " -Dm755 -t " ${ PI_HOLE_INSTALL_DIR } " gravity.sh
install -o " ${ USER } " -Dm755 -t " ${ PI_HOLE_INSTALL_DIR } " ./advanced/Scripts/*.sh
install -o " ${ USER } " -Dm755 -t " ${ PI_HOLE_INSTALL_DIR } " ./automated\ install/uninstall.sh
install -o " ${ USER } " -Dm755 -t " ${ PI_HOLE_INSTALL_DIR } " ./advanced/Scripts/COL_TABLE
2019-08-24 11:33:32 +00:00
install -o " ${ USER } " -Dm755 -t " ${ PI_HOLE_BIN_DIR } " pihole
2018-07-06 04:48:36 +00:00
install -Dm644 ./advanced/bash-completion/pihole /etc/bash_completion.d/pihole
2018-10-31 18:14:05 +00:00
printf "%b %b %s\\n" " ${ OVER } " " ${ TICK } " " ${ str } "
2018-07-30 21:02:18 +00:00
2018-07-06 04:48:36 +00:00
else
2021-03-18 01:35:27 +00:00
# Otherwise, show an error and exit
2018-10-31 18:14:05 +00:00
printf "%b %b %s\\n" " ${ OVER } " " ${ CROSS } " " ${ str } "
2018-10-31 21:35:55 +00:00
printf "\\t\\t%bError: Local repo %s not found, exiting installer%b\\n" " ${ COL_LIGHT_RED } " " ${ PI_HOLE_LOCAL_REPO } " " ${ COL_NC } "
2018-07-29 16:06:51 +00:00
return 1
2018-07-06 04:48:36 +00:00
fi
2015-12-03 17:25:13 +00:00
}
2017-07-07 00:25:56 +00:00
# Install the configs from PI_HOLE_LOCAL_REPO to their various locations
2016-01-27 06:11:38 +00:00
installConfigs( ) {
2018-10-31 18:14:05 +00:00
printf "\\n %b Installing configs from %s...\\n" " ${ INFO } " " ${ PI_HOLE_LOCAL_REPO } "
2018-07-06 04:48:36 +00:00
# Make sure Pi-hole's config files are in place
version_check_dnsmasq
2019-01-13 14:51:30 +00:00
# Install list of DNS servers
# Format: Name;Primary IPv4;Secondary IPv4;Primary IPv6;Secondary IPv6
# Some values may be empty (for example: DNS servers without IPv6 support)
echo " ${ DNS_SERVERS } " > " ${ PI_HOLE_CONFIG_DIR } /dns-servers.conf "
2019-05-04 22:32:10 +00:00
chmod 644 " ${ PI_HOLE_CONFIG_DIR } /dns-servers.conf "
2019-01-13 14:51:30 +00:00
2022-01-04 19:09:30 +00:00
# Install template file if it does not exist
2019-02-05 09:33:31 +00:00
if [ [ ! -r " ${ PI_HOLE_CONFIG_DIR } /pihole-FTL.conf " ] ] ; then
install -d -m 0755 ${ PI_HOLE_CONFIG_DIR }
2022-01-04 19:09:30 +00:00
if ! install -T -o pihole -m 664 " ${ PI_HOLE_LOCAL_REPO } /advanced/Templates/pihole-FTL.conf " " ${ PI_HOLE_CONFIG_DIR } /pihole-FTL.conf " & >/dev/null; then
2022-07-06 10:29:03 +00:00
printf " %b Error: Unable to initialize configuration file %s/pihole-FTL.conf\\n" " ${ COL_LIGHT_RED } " " ${ PI_HOLE_CONFIG_DIR } "
2018-07-30 21:02:18 +00:00
return 1
fi
fi
2020-07-24 18:32:32 +00:00
# Install empty custom.list file if it does not exist
if [ [ ! -r " ${ PI_HOLE_CONFIG_DIR } /custom.list " ] ] ; then
2020-07-26 19:31:11 +00:00
if ! install -o root -m 644 /dev/null " ${ PI_HOLE_CONFIG_DIR } /custom.list " & >/dev/null; then
2022-07-06 10:29:03 +00:00
printf " %b Error: Unable to initialize configuration file %s/custom.list\\n" " ${ COL_LIGHT_RED } " " ${ PI_HOLE_CONFIG_DIR } "
2020-07-24 18:32:32 +00:00
return 1
fi
fi
2020-07-26 10:27:55 +00:00
2022-05-19 04:17:46 +00:00
# Install pihole-FTL.service
install -T -m 0755 " ${ PI_HOLE_LOCAL_REPO } /advanced/Templates/pihole-FTL.service " "/etc/init.d/pihole-FTL"
2018-07-06 04:48:36 +00:00
# If the user chose to install the dashboard,
if [ [ " ${ INSTALL_WEB_SERVER } " = = true ] ] ; then
# and if the Web server conf directory does not exist,
if [ [ ! -d "/etc/lighttpd" ] ] ; then
2019-05-03 10:27:56 +00:00
# make it and set the owners
install -d -m 755 -o " ${ USER } " -g root /etc/lighttpd
2018-07-06 04:48:36 +00:00
# Otherwise, if the config file already exists
2021-10-05 15:40:12 +00:00
elif [ [ -f " ${ lighttpdConfig } " ] ] ; then
2018-07-06 04:48:36 +00:00
# back up the original
2021-10-05 15:40:12 +00:00
mv " ${ lighttpdConfig } " { ,.orig}
2018-07-06 04:48:36 +00:00
fi
# and copy in the config file Pi-hole needs
2021-10-05 15:40:12 +00:00
install -D -m 644 -T ${ PI_HOLE_LOCAL_REPO } /advanced/${ LIGHTTPD_CFG } " ${ lighttpdConfig } "
2018-09-07 16:09:37 +00:00
# Make sure the external.conf file exists, as lighttpd v1.4.50 crashes without it
2022-03-31 19:03:17 +00:00
if [ ! -f /etc/lighttpd/external.conf ] ; then
2022-03-31 21:32:07 +00:00
install -m 644 /dev/null /etc/lighttpd/external.conf
2022-03-31 19:03:17 +00:00
fi
2021-03-18 01:35:27 +00:00
# If there is a custom block page in the html/pihole directory, replace 404 handler in lighttpd config
2019-02-05 09:17:53 +00:00
if [ [ -f " ${ PI_HOLE_BLOCKPAGE_DIR } /custom.php " ] ] ; then
2021-12-28 18:32:06 +00:00
sed -i 's/^\(server\.error-handler-404\s*=\s*\).*$/\1"\/pihole\/custom\.php"/' " ${ lighttpdConfig } "
2018-07-06 04:48:36 +00:00
fi
# Make the directories if they do not exist and set the owners
2020-04-03 17:05:59 +00:00
mkdir -p /run/lighttpd
chown ${ LIGHTTPD_USER } :${ LIGHTTPD_GROUP } /run/lighttpd
2018-07-06 04:48:36 +00:00
mkdir -p /var/cache/lighttpd/compress
chown ${ LIGHTTPD_USER } :${ LIGHTTPD_GROUP } /var/cache/lighttpd/compress
mkdir -p /var/cache/lighttpd/uploads
chown ${ LIGHTTPD_USER } :${ LIGHTTPD_GROUP } /var/cache/lighttpd/uploads
fi
2015-12-03 17:25:13 +00:00
}
2018-06-01 23:45:04 +00:00
install_manpage( ) {
2018-07-06 04:48:36 +00:00
# Copy Pi-hole man pages and call mandb to update man page database
# Default location for man files for /usr/local/bin is /usr/local/share/man
# on lightweight systems may not be present, so check before copying.
2018-10-31 18:14:05 +00:00
printf " %b Testing man page installation" " ${ INFO } "
2018-11-01 19:00:08 +00:00
if ! is_command mandb ; then
2018-07-06 04:48:36 +00:00
# if mandb is not present, no manpage support
2018-10-31 18:14:05 +00:00
printf "%b %b man not installed\\n" " ${ OVER } " " ${ INFO } "
2018-07-06 04:48:36 +00:00
return
elif [ [ ! -d "/usr/local/share/man" ] ] ; then
# appropriate directory for Pi-hole's man page is not present
2018-10-31 18:14:05 +00:00
printf "%b %b man pages not installed\\n" " ${ OVER } " " ${ INFO } "
2018-07-06 04:48:36 +00:00
return
fi
if [ [ ! -d "/usr/local/share/man/man8" ] ] ; then
# if not present, create man8 directory
2019-05-03 10:27:56 +00:00
install -d -m 755 /usr/local/share/man/man8
2018-07-06 04:48:36 +00:00
fi
if [ [ ! -d "/usr/local/share/man/man5" ] ] ; then
2019-04-30 20:41:12 +00:00
# if not present, create man5 directory
2019-05-03 10:27:56 +00:00
install -d -m 755 /usr/local/share/man/man5
2018-07-06 04:48:36 +00:00
fi
# Testing complete, copy the files & update the man db
2019-05-03 10:27:56 +00:00
install -D -m 644 -T ${ PI_HOLE_LOCAL_REPO } /manpages/pihole.8 /usr/local/share/man/man8/pihole.8
install -D -m 644 -T ${ PI_HOLE_LOCAL_REPO } /manpages/pihole-FTL.8 /usr/local/share/man/man8/pihole-FTL.8
2021-12-29 05:52:17 +00:00
2022-02-04 20:11:54 +00:00
# remove previously installed "pihole-FTL.conf.5" man page
2021-12-29 05:52:17 +00:00
if [ [ -f "/usr/local/share/man/man5/pihole-FTL.conf.5" ] ] ; then
rm /usr/local/share/man/man5/pihole-FTL.conf.5
fi
2018-07-06 04:48:36 +00:00
if mandb -q & >/dev/null; then
# Updated successfully
2018-10-31 18:14:05 +00:00
printf "%b %b man pages installed and database updated\\n" " ${ OVER } " " ${ TICK } "
2018-07-06 04:48:36 +00:00
return
else
# Something is wrong with the system's man installation, clean up
# our files, (leave everything how we found it).
2021-12-29 05:52:17 +00:00
rm /usr/local/share/man/man8/pihole.8 /usr/local/share/man/man8/pihole-FTL.8
2018-10-31 18:14:05 +00:00
printf "%b %b man page db not updated, man pages not installed\\n" " ${ OVER } " " ${ CROSS } "
2018-07-06 04:48:36 +00:00
fi
2018-06-01 23:45:04 +00:00
}
2016-10-10 08:25:11 +00:00
stop_service( ) {
2018-07-06 04:48:36 +00:00
# Stop service passed in as argument.
# Can softfail, as process may not be installed when this is called
local str = " Stopping ${ 1 } service "
2018-10-31 18:14:05 +00:00
printf " %b %s..." " ${ INFO } " " ${ str } "
2018-11-01 19:00:08 +00:00
if is_command systemctl ; then
2018-07-06 04:48:36 +00:00
systemctl stop " ${ 1 } " & > /dev/null || true
else
service " ${ 1 } " stop & > /dev/null || true
fi
2018-10-31 18:14:05 +00:00
printf "%b %b %s...\\n" " ${ OVER } " " ${ TICK } " " ${ str } "
2016-10-10 13:06:34 +00:00
}
2017-07-07 00:25:56 +00:00
# Start/Restart service passed in as argument
2019-01-24 08:37:03 +00:00
restart_service( ) {
2018-07-06 04:48:36 +00:00
# Local, named variables
2019-02-02 07:00:58 +00:00
local str = " Restarting ${ 1 } service "
2018-10-31 18:14:05 +00:00
printf " %b %s..." " ${ INFO } " " ${ str } "
2018-07-06 04:48:36 +00:00
# If systemctl exists,
2018-11-01 19:00:08 +00:00
if is_command systemctl ; then
2018-07-06 04:48:36 +00:00
# use that to restart the service
systemctl restart " ${ 1 } " & > /dev/null
else
2021-03-18 01:35:27 +00:00
# Otherwise, fall back to the service command
2018-07-06 04:48:36 +00:00
service " ${ 1 } " restart & > /dev/null
fi
2018-10-31 18:14:05 +00:00
printf "%b %b %s...\\n" " ${ OVER } " " ${ TICK } " " ${ str } "
2016-10-10 13:06:34 +00:00
}
2017-07-07 00:25:56 +00:00
# Enable service so that it will start with next reboot
2016-10-10 13:06:34 +00:00
enable_service( ) {
2018-07-06 04:48:36 +00:00
# Local, named variables
local str = " Enabling ${ 1 } service to start on reboot "
2018-10-31 18:14:05 +00:00
printf " %b %s..." " ${ INFO } " " ${ str } "
2018-07-06 04:48:36 +00:00
# If systemctl exists,
2018-11-01 19:00:08 +00:00
if is_command systemctl ; then
2018-07-06 04:48:36 +00:00
# use that to enable the service
systemctl enable " ${ 1 } " & > /dev/null
else
2021-03-18 01:35:27 +00:00
# Otherwise, use update-rc.d to accomplish this
2018-07-06 04:48:36 +00:00
update-rc.d " ${ 1 } " defaults & > /dev/null
fi
2018-10-31 18:14:05 +00:00
printf "%b %b %s...\\n" " ${ OVER } " " ${ TICK } " " ${ str } "
2015-12-03 17:25:13 +00:00
}
2016-10-10 05:05:06 +00:00
2018-03-06 18:44:57 +00:00
# Disable service so that it will not with next reboot
disable_service( ) {
2018-07-06 04:48:36 +00:00
# Local, named variables
local str = " Disabling ${ 1 } service "
2018-10-31 18:14:05 +00:00
printf " %b %s..." " ${ INFO } " " ${ str } "
2018-07-06 04:48:36 +00:00
# If systemctl exists,
2018-11-01 19:00:08 +00:00
if is_command systemctl ; then
2018-07-06 04:48:36 +00:00
# use that to disable the service
systemctl disable " ${ 1 } " & > /dev/null
else
2021-03-18 01:35:27 +00:00
# Otherwise, use update-rc.d to accomplish this
2018-07-06 04:48:36 +00:00
update-rc.d " ${ 1 } " disable & > /dev/null
fi
2018-10-31 18:14:05 +00:00
printf "%b %b %s...\\n" " ${ OVER } " " ${ TICK } " " ${ str } "
2018-03-06 18:44:57 +00:00
}
2018-03-07 23:31:28 +00:00
check_service_active( ) {
# If systemctl exists,
2018-11-01 19:00:08 +00:00
if is_command systemctl ; then
2018-07-06 04:48:36 +00:00
# use that to check the status of the service
2018-07-29 00:50:04 +00:00
systemctl is-enabled " ${ 1 } " & > /dev/null
2018-07-06 04:48:36 +00:00
else
2021-03-18 01:35:27 +00:00
# Otherwise, fall back to service command
2018-07-29 00:50:04 +00:00
service " ${ 1 } " status & > /dev/null
2018-07-06 04:48:36 +00:00
fi
2018-03-07 23:31:28 +00:00
}
2018-07-06 04:48:36 +00:00
# Systemd-resolved's DNSStubListener and dnsmasq can't share port 53.
2018-05-15 10:23:36 +00:00
disable_resolved_stublistener( ) {
2018-10-31 18:14:05 +00:00
printf " %b Testing if systemd-resolved is enabled\\n" " ${ INFO } "
2018-07-26 23:27:06 +00:00
# Check if Systemd-resolved's DNSStubListener is enabled and active on port 53
if check_service_active "systemd-resolved" ; then
2018-07-06 04:48:36 +00:00
# Check if DNSStubListener is enabled
2022-07-06 10:29:03 +00:00
printf " %b %b Testing if systemd-resolved DNSStub-Listener is active" " ${ OVER } " " ${ INFO } "
2018-07-06 04:48:36 +00:00
if ( grep -E '#?DNSStubListener=yes' /etc/systemd/resolved.conf & > /dev/null ) ; then
# Disable the DNSStubListener to unbind it from port 53
# Note that this breaks dns functionality on host until dnsmasq/ftl are up and running
2018-10-31 18:14:05 +00:00
printf "%b %b Disabling systemd-resolved DNSStubListener" " ${ OVER } " " ${ TICK } "
2018-07-06 04:48:36 +00:00
# Make a backup of the original /etc/systemd/resolved.conf
# (This will need to be restored on uninstallation)
2018-07-26 23:27:06 +00:00
sed -r -i.orig 's/#?DNSStubListener=yes/DNSStubListener=no/g' /etc/systemd/resolved.conf
2018-10-31 18:14:05 +00:00
printf " and restarting systemd-resolved\\n"
2018-07-26 23:27:06 +00:00
systemctl reload-or-restart systemd-resolved
2018-07-06 04:48:36 +00:00
else
2018-10-31 18:14:05 +00:00
printf "%b %b Systemd-resolved does not need to be restarted\\n" " ${ OVER } " " ${ INFO } "
2018-07-06 04:48:36 +00:00
fi
2018-05-16 21:44:07 +00:00
else
2018-10-31 18:14:05 +00:00
printf "%b %b Systemd-resolved is not enabled\\n" " ${ OVER } " " ${ INFO } "
2018-05-15 10:23:36 +00:00
fi
}
2017-01-28 23:44:31 +00:00
update_package_cache( ) {
2018-07-06 04:48:36 +00:00
# Update package cache on apt based OSes. Do this every time since
# it's quick and packages can be updated at any time.
# Local, named variables
local str = "Update local cache of available packages"
2018-10-31 18:14:05 +00:00
printf " %b %s..." " ${ INFO } " " ${ str } "
2018-07-06 04:48:36 +00:00
# Create a command from the package cache variable
if eval " ${ UPDATE_PKG_CACHE } " & > /dev/null; then
2018-10-31 18:14:05 +00:00
printf "%b %b %s\\n" " ${ OVER } " " ${ TICK } " " ${ str } "
2018-07-06 04:48:36 +00:00
else
2021-03-18 01:35:27 +00:00
# Otherwise, show an error and exit
2021-11-04 22:55:16 +00:00
# In case we used apt-get and apt is also available, we use this as recommendation as we have seen it
# gives more user-friendly (interactive) advice
if [ [ ${ PKG_MANAGER } = = "apt-get" ] ] && is_command apt ; then
UPDATE_PKG_CACHE = "apt update"
fi
2018-10-31 18:14:05 +00:00
printf "%b %b %s\\n" " ${ OVER } " " ${ CROSS } " " ${ str } "
2022-07-06 10:29:03 +00:00
printf " %b Error: Unable to update package cache. Please try \"%s\"%b\\n" " ${ COL_LIGHT_RED } " " sudo ${ UPDATE_PKG_CACHE } " " ${ COL_NC } "
2018-07-06 04:48:36 +00:00
return 1
fi
2016-10-10 09:24:03 +00:00
}
2017-07-07 00:25:56 +00:00
# Let user know if they have outdated packages on their system and
# advise them to run a package update at soonest possible.
2016-10-20 02:47:45 +00:00
notify_package_updates_available( ) {
2018-07-06 04:48:36 +00:00
# Local, named variables
local str = " Checking ${ PKG_MANAGER } for upgraded packages "
2018-10-31 18:14:05 +00:00
printf "\\n %b %s..." " ${ INFO } " " ${ str } "
2018-07-06 04:48:36 +00:00
# Store the list of packages in a variable
updatesToInstall = $( eval " ${ PKG_COUNT } " )
2017-07-26 17:00:08 +00:00
2018-07-06 04:48:36 +00:00
if [ [ -d " /lib/modules/ $( uname -r) " ] ] ; then
if [ [ " ${ updatesToInstall } " -eq 0 ] ] ; then
2018-10-31 18:14:05 +00:00
printf "%b %b %s... up to date!\\n\\n" " ${ OVER } " " ${ TICK } " " ${ str } "
2018-07-06 04:48:36 +00:00
else
2018-10-31 18:14:05 +00:00
printf "%b %b %s... %s updates available\\n" " ${ OVER } " " ${ TICK } " " ${ str } " " ${ updatesToInstall } "
printf " %b %bIt is recommended to update your OS after installing the Pi-hole!%b\\n\\n" " ${ INFO } " " ${ COL_LIGHT_GREEN } " " ${ COL_NC } "
2018-07-06 04:48:36 +00:00
fi
2016-12-29 09:34:49 +00:00
else
2018-10-31 18:14:05 +00:00
printf "%b %b %s\\n" " ${ OVER } " " ${ CROSS } " " ${ str } "
printf " Kernel update detected. If the install fails, please reboot and try again\\n"
2018-07-06 04:48:36 +00:00
fi
2016-10-10 05:00:23 +00:00
}
2016-10-20 02:47:45 +00:00
install_dependent_packages( ) {
2018-07-06 04:48:36 +00:00
# Install packages passed in via argument array
# No spinner - conflicts with set -e
declare -a installArray
# Debian based package install - debconf will download the entire package list
# so we just create an array of packages not currently installed to cut down on the
# amount of download traffic.
# NOTE: We may be able to use this installArray in the future to create a list of package that were
# installed by us, and remove only the installed packages, and not the entire list.
2019-10-13 17:44:15 +00:00
if is_command apt-get ; then
2021-03-18 01:35:27 +00:00
# For each package, check if it's already installed (and if so, don't add it to the installArray)
2019-05-10 14:37:38 +00:00
for i in " $@ " ; do
2018-10-31 18:14:05 +00:00
printf " %b Checking for %s..." " ${ INFO } " " ${ i } "
2018-07-06 04:48:36 +00:00
if dpkg-query -W -f= '${Status}' " ${ i } " 2>/dev/null | grep "ok installed" & > /dev/null; then
2018-10-31 18:14:05 +00:00
printf "%b %b Checking for %s\\n" " ${ OVER } " " ${ TICK } " " ${ i } "
2018-07-06 04:48:36 +00:00
else
2019-10-15 02:16:40 +00:00
printf "%b %b Checking for %s (will be installed)\\n" " ${ OVER } " " ${ INFO } " " ${ i } "
2018-07-06 04:48:36 +00:00
installArray += ( " ${ i } " )
fi
done
2021-03-18 01:35:27 +00:00
# If there's anything to install, install everything in the list.
2018-07-06 04:48:36 +00:00
if [ [ " ${# installArray [@] } " -gt 0 ] ] ; then
test_dpkg_lock
2021-04-07 19:53:52 +00:00
# Running apt-get install with minimal output can cause some issues with
# requiring user input (e.g password for phpmyadmin see #218)
2020-01-01 18:41:33 +00:00
printf " %b Processing %s install(s) for: %s, please wait...\\n" " ${ INFO } " " ${ PKG_MANAGER } " " ${ installArray [*] } "
printf '%*s\n' " $columns " '' | tr " " -;
2019-10-13 17:44:15 +00:00
" ${ PKG_INSTALL [@] } " " ${ installArray [@] } "
2020-01-01 18:41:33 +00:00
printf '%*s\n' " $columns " '' | tr " " -;
2018-07-06 04:48:36 +00:00
return
fi
2018-10-31 18:14:05 +00:00
printf "\\n"
2018-07-06 04:48:36 +00:00
return 0
fi
# Install Fedora/CentOS packages
2019-05-10 14:37:38 +00:00
for i in " $@ " ; do
2021-11-25 06:41:40 +00:00
# For each package, check if it's already installed (and if so, don't add it to the installArray)
2018-10-31 18:14:05 +00:00
printf " %b Checking for %s..." " ${ INFO } " " ${ i } "
2019-05-10 14:13:23 +00:00
if " ${ PKG_MANAGER } " -q list installed " ${ i } " & > /dev/null; then
2019-10-13 20:35:38 +00:00
printf "%b %b Checking for %s\\n" " ${ OVER } " " ${ TICK } " " ${ i } "
2018-07-06 04:48:36 +00:00
else
2019-10-15 02:16:40 +00:00
printf "%b %b Checking for %s (will be installed)\\n" " ${ OVER } " " ${ INFO } " " ${ i } "
2018-07-06 04:48:36 +00:00
installArray += ( " ${ i } " )
fi
2016-12-21 00:47:43 +00:00
done
2021-03-18 01:35:27 +00:00
# If there's anything to install, install everything in the list.
2017-07-26 17:00:08 +00:00
if [ [ " ${# installArray [@] } " -gt 0 ] ] ; then
2020-01-01 18:41:33 +00:00
printf " %b Processing %s install(s) for: %s, please wait...\\n" " ${ INFO } " " ${ PKG_MANAGER } " " ${ installArray [*] } "
printf '%*s\n' " $columns " '' | tr " " -;
" ${ PKG_INSTALL [@] } " " ${ installArray [@] } "
printf '%*s\n' " $columns " '' | tr " " -;
2018-07-06 04:48:36 +00:00
return
2016-12-22 07:38:31 +00:00
fi
2018-10-31 18:14:05 +00:00
printf "\\n"
2018-07-06 04:48:36 +00:00
return 0
2015-12-03 17:25:13 +00:00
}
2017-07-07 00:25:56 +00:00
# Install the Web interface dashboard
2016-01-27 06:11:38 +00:00
installPiholeWeb( ) {
2018-10-31 18:14:05 +00:00
printf "\\n %b Installing blocking page...\\n" " ${ INFO } "
2018-07-06 04:48:36 +00:00
local str = "Creating directory for blocking page, and copying files"
2018-10-31 18:14:05 +00:00
printf " %b %s..." " ${ INFO } " " ${ str } "
2021-03-18 01:35:27 +00:00
# Install the directory,
2019-02-05 09:23:23 +00:00
install -d -m 0755 ${ PI_HOLE_BLOCKPAGE_DIR }
2018-07-06 04:48:36 +00:00
# and the blockpage
2019-05-03 10:27:56 +00:00
install -D -m 644 ${ PI_HOLE_LOCAL_REPO } /advanced/{ index,blockingpage} .* ${ PI_HOLE_BLOCKPAGE_DIR } /
2018-07-06 04:48:36 +00:00
# Remove superseded file
2019-02-05 09:17:53 +00:00
if [ [ -e " ${ PI_HOLE_BLOCKPAGE_DIR } /index.js " ] ] ; then
rm " ${ PI_HOLE_BLOCKPAGE_DIR } /index.js "
2018-07-06 04:48:36 +00:00
fi
2018-10-31 18:14:05 +00:00
printf "%b %b %s\\n" " ${ OVER } " " ${ TICK } " " ${ str } "
2018-07-06 04:48:36 +00:00
local str = "Backing up index.lighttpd.html"
2018-10-31 18:14:05 +00:00
printf " %b %s..." " ${ INFO } " " ${ str } "
2018-07-06 04:48:36 +00:00
# If the default index file exists,
2019-02-05 09:30:43 +00:00
if [ [ -f " ${ webroot } /index.lighttpd.html " ] ] ; then
2018-07-06 04:48:36 +00:00
# back it up
2019-02-05 09:30:43 +00:00
mv ${ webroot } /index.lighttpd.html ${ webroot } /index.lighttpd.orig
2018-10-31 18:14:05 +00:00
printf "%b %b %s\\n" " ${ OVER } " " ${ TICK } " " ${ str } "
2018-07-06 04:48:36 +00:00
else
2021-03-18 01:35:27 +00:00
# Otherwise, don't do anything
2019-09-17 19:59:48 +00:00
printf "%b %b %s\\n" " ${ OVER } " " ${ INFO } " " ${ str } "
2018-10-31 18:14:05 +00:00
printf " No default index.lighttpd.html file found... not backing up\\n"
2018-07-06 04:48:36 +00:00
fi
# Install Sudoers file
local str = "Installing sudoer file"
2018-10-31 18:14:05 +00:00
printf "\\n %b %s..." " ${ INFO } " " ${ str } "
2021-03-18 01:35:27 +00:00
# Make the .d directory if it doesn't exist,
2019-05-04 22:32:10 +00:00
install -d -m 755 /etc/sudoers.d/
2018-07-06 04:48:36 +00:00
# and copy in the pihole sudoers file
2019-02-05 12:45:52 +00:00
install -m 0640 ${ PI_HOLE_LOCAL_REPO } /advanced/Templates/pihole.sudo /etc/sudoers.d/pihole
2018-07-06 04:48:36 +00:00
# Add lighttpd user (OS dependent) to sudoers file
2019-08-24 11:33:32 +00:00
echo " ${ LIGHTTPD_USER } ALL=NOPASSWD: ${ PI_HOLE_BIN_DIR } /pihole " >> /etc/sudoers.d/pihole
2018-07-06 04:48:36 +00:00
# If the Web server user is lighttpd,
if [ [ " $LIGHTTPD_USER " = = "lighttpd" ] ] ; then
# Allow executing pihole via sudo with Fedora
2019-08-24 11:33:32 +00:00
# Usually /usr/local/bin ${PI_HOLE_BIN_DIR} is not permitted as directory for sudoable programs
echo " Defaults secure_path = /sbin:/bin:/usr/sbin:/usr/bin: ${ PI_HOLE_BIN_DIR } " >> /etc/sudoers.d/pihole
2018-07-06 04:48:36 +00:00
fi
# Set the strict permissions on the file
chmod 0440 /etc/sudoers.d/pihole
2018-10-31 18:14:05 +00:00
printf "%b %b %s\\n" " ${ OVER } " " ${ TICK } " " ${ str } "
2015-12-03 17:25:13 +00:00
}
2017-07-07 00:25:56 +00:00
# Installs a cron file
2016-01-27 06:11:38 +00:00
installCron( ) {
2018-07-06 04:48:36 +00:00
# Install the cron job
local str = "Installing latest Cron script"
2018-10-31 18:14:05 +00:00
printf "\\n %b %s..." " ${ INFO } " " ${ str } "
2018-07-06 04:48:36 +00:00
# Copy the cron file over from the local repo
2019-04-30 20:41:12 +00:00
# File must not be world or group writeable and must be owned by root
2019-05-03 10:27:56 +00:00
install -D -m 644 -T -o root -g root ${ PI_HOLE_LOCAL_REPO } /advanced/Templates/pihole.cron /etc/cron.d/pihole
2018-07-06 04:48:36 +00:00
# Randomize gravity update time
sed -i " s/59 1 / $(( 1 + RANDOM % 58 )) $(( 3 + RANDOM % 2 )) / " /etc/cron.d/pihole
# Randomize update checker time
sed -i " s/59 17/ $(( 1 + RANDOM % 58 )) $(( 12 + RANDOM % 8 )) / " /etc/cron.d/pihole
2018-10-31 18:14:05 +00:00
printf "%b %b %s\\n" " ${ OVER } " " ${ TICK } " " ${ str } "
2015-12-03 17:25:13 +00:00
}
2017-07-07 00:25:56 +00:00
# Gravity is a very important script as it aggregates all of the domains into a single HOSTS formatted list,
# which is what Pi-hole needs to begin blocking ads
2016-01-27 06:11:38 +00:00
runGravity( ) {
2018-07-06 04:48:36 +00:00
# Run gravity in the current shell
{ /opt/pihole/gravity.sh --force; }
2016-01-19 22:52:29 +00:00
}
2017-07-07 00:25:56 +00:00
# Check if the pihole user exists and create if it does not
2016-10-20 02:47:45 +00:00
create_pihole_user( ) {
2018-07-06 04:48:36 +00:00
local str = "Checking for user 'pihole'"
2018-10-31 18:14:05 +00:00
printf " %b %s..." " ${ INFO } " " ${ str } "
2021-03-18 01:35:27 +00:00
# If the pihole user exists,
2018-07-06 04:48:36 +00:00
if id -u pihole & > /dev/null; then
2021-03-18 01:35:27 +00:00
# and if the pihole group exists,
2020-04-03 18:07:44 +00:00
if getent group pihole > /dev/null 2>& 1; then
2021-03-18 01:35:27 +00:00
# succeed
2020-04-03 10:06:59 +00:00
printf "%b %b %s\\n" " ${ OVER } " " ${ TICK } " " ${ str } "
else
local str = "Checking for group 'pihole'"
printf " %b %s..." " ${ INFO } " " ${ str } "
local str = "Creating group 'pihole'"
# if group can be created
if groupadd pihole; then
printf "%b %b %s\\n" " ${ OVER } " " ${ TICK } " " ${ str } "
local str = "Adding user 'pihole' to group 'pihole'"
printf " %b %s..." " ${ INFO } " " ${ str } "
# if pihole user can be added to group pihole
if usermod -g pihole pihole; then
printf "%b %b %s\\n" " ${ OVER } " " ${ TICK } " " ${ str } "
else
printf "%b %b %s\\n" " ${ OVER } " " ${ CROSS } " " ${ str } "
fi
else
printf "%b %b %s\\n" " ${ OVER } " " ${ CROSS } " " ${ str } "
fi
fi
2018-07-06 04:48:36 +00:00
else
2021-03-18 01:35:27 +00:00
# If the pihole user doesn't exist,
2018-10-31 18:14:05 +00:00
printf "%b %b %s" " ${ OVER } " " ${ CROSS } " " ${ str } "
2018-07-06 04:48:36 +00:00
local str = "Creating user 'pihole'"
2018-10-31 18:14:05 +00:00
printf "%b %b %s..." " ${ OVER } " " ${ INFO } " " ${ str } "
2021-03-18 01:35:27 +00:00
# create her with the useradd command,
2020-04-03 18:07:44 +00:00
if getent group pihole > /dev/null 2>& 1; then
2021-03-18 01:35:27 +00:00
# then add her to the pihole group (as it already exists)
2020-04-03 10:06:59 +00:00
if useradd -r --no-user-group -g pihole -s /usr/sbin/nologin pihole; then
printf "%b %b %s\\n" " ${ OVER } " " ${ TICK } " " ${ str } "
else
printf "%b %b %s\\n" " ${ OVER } " " ${ CROSS } " " ${ str } "
fi
2018-10-12 17:05:44 +00:00
else
2020-04-03 10:06:59 +00:00
# add user pihole with default group settings
if useradd -r -s /usr/sbin/nologin pihole; then
printf "%b %b %s\\n" " ${ OVER } " " ${ TICK } " " ${ str } "
else
printf "%b %b %s\\n" " ${ OVER } " " ${ CROSS } " " ${ str } "
fi
2018-10-12 17:05:44 +00:00
fi
2018-07-06 04:48:36 +00:00
fi
2016-01-25 05:28:53 +00:00
}
2016-01-20 23:34:18 +00:00
2021-03-18 01:35:27 +00:00
# This function saves any changes to the setup variables into the setupvars.conf file for future runs
2016-08-26 21:45:38 +00:00
finalExports( ) {
2018-07-06 04:48:36 +00:00
# If the setup variable file exists,
if [ [ -e " ${ setupVars } " ] ] ; then
# update the variables in the file
2022-01-09 05:56:24 +00:00
sed -i.update.bak '/PIHOLE_INTERFACE/d;/PIHOLE_DNS_1\b/d;/PIHOLE_DNS_2\b/d;/QUERY_LOGGING/d;/INSTALL_WEB_SERVER/d;/INSTALL_WEB_INTERFACE/d;/LIGHTTPD_ENABLED/d;/CACHE_SIZE/d;/DNS_FQDN_REQUIRED/d;/DNS_BOGUS_PRIV/d;/DNSMASQ_LISTENING/d;' " ${ setupVars } "
2018-07-06 04:48:36 +00:00
fi
# echo the information to the user
2016-10-10 09:57:04 +00:00
{
2021-11-25 06:41:40 +00:00
echo " PIHOLE_INTERFACE= ${ PIHOLE_INTERFACE } "
echo " PIHOLE_DNS_1= ${ PIHOLE_DNS_1 } "
echo " PIHOLE_DNS_2= ${ PIHOLE_DNS_2 } "
echo " QUERY_LOGGING= ${ QUERY_LOGGING } "
echo " INSTALL_WEB_SERVER= ${ INSTALL_WEB_SERVER } "
echo " INSTALL_WEB_INTERFACE= ${ INSTALL_WEB_INTERFACE } "
echo " LIGHTTPD_ENABLED= ${ LIGHTTPD_ENABLED } "
echo " CACHE_SIZE= ${ CACHE_SIZE } "
echo " DNS_FQDN_REQUIRED= ${ DNS_FQDN_REQUIRED :- true } "
echo " DNS_BOGUS_PRIV= ${ DNS_BOGUS_PRIV :- true } "
2022-01-04 08:40:07 +00:00
echo " DNSMASQ_LISTENING= ${ DNSMASQ_LISTENING :- local } "
2016-10-10 12:45:37 +00:00
} >> " ${ setupVars } "
2019-05-01 09:20:26 +00:00
chmod 644 " ${ setupVars } "
2016-12-28 16:31:55 +00:00
2018-08-20 23:04:58 +00:00
# Set the privacy level
sed -i '/PRIVACYLEVEL/d' " ${ PI_HOLE_CONFIG_DIR } /pihole-FTL.conf "
echo " PRIVACYLEVEL= ${ PRIVACY_LEVEL } " >> " ${ PI_HOLE_CONFIG_DIR } /pihole-FTL.conf "
2018-07-06 04:48:36 +00:00
# Bring in the current settings and the functions to manipulate them
source " ${ setupVars } "
source " ${ PI_HOLE_LOCAL_REPO } /advanced/Scripts/webpage.sh "
2017-01-02 09:50:59 +00:00
2018-07-06 04:48:36 +00:00
# Look for DNS server settings which would have to be reapplied
ProcessDNSSettings
2017-01-02 09:50:59 +00:00
2018-07-06 04:48:36 +00:00
# Look for DHCP server settings which would have to be reapplied
ProcessDHCPSettings
2015-11-07 18:07:50 +00:00
}
2017-07-07 00:25:56 +00:00
# Install the logrotate script
2017-01-27 13:16:24 +00:00
installLogrotate( ) {
2018-07-06 04:48:36 +00:00
local str = "Installing latest logrotate script"
2021-06-20 18:18:33 +00:00
local target = /etc/pihole/logrotate
2018-10-31 21:35:55 +00:00
printf "\\n %b %s..." " ${ INFO } " " ${ str } "
2021-06-20 18:18:33 +00:00
if [ [ -f ${ target } ] ] ; then
printf "\\n\\t%b Existing logrotate file found. No changes made.\\n" " ${ INFO } "
# Return value isn't that important, using 2 to indicate that it's not a fatal error but
2021-08-03 20:48:26 +00:00
# the function did not complete.
2021-06-20 18:18:33 +00:00
return 2
fi
2018-07-06 04:48:36 +00:00
# Copy the file over from the local repo
2021-06-20 18:19:19 +00:00
install -D -m 644 -T " ${ PI_HOLE_LOCAL_REPO } " /advanced/Templates/logrotate ${ target }
2018-07-06 04:48:36 +00:00
# Different operating systems have different user / group
# settings for logrotate that makes it impossible to create
# a static logrotate file that will work with e.g.
# Rasbian and Ubuntu at the same time. Hence, we have to
# customize the logrotate script here in order to reflect
# the local properties of the /var/log directory
logusergroup = " $( stat -c '%U %G' /var/log) "
2021-03-18 01:35:27 +00:00
# If there is a usergroup for log rotation,
2021-06-20 18:19:19 +00:00
if [ [ -n " ${ logusergroup } " ] ] ; then
2021-03-18 01:35:27 +00:00
# replace the line in the logrotate script with that usergroup.
2021-06-20 18:18:33 +00:00
sed -i " s/# su #/su ${ logusergroup } /g; " ${ target }
2018-07-06 04:48:36 +00:00
fi
2018-10-31 18:14:05 +00:00
printf "%b %b %s\\n" " ${ OVER } " " ${ TICK } " " ${ str } "
2017-01-27 13:16:24 +00:00
}
2017-07-07 00:25:56 +00:00
# Install base files and web interface
2016-08-19 21:45:24 +00:00
installPihole( ) {
2018-07-06 04:48:36 +00:00
# If the user wants to install the Web interface,
if [ [ " ${ INSTALL_WEB_INTERFACE } " = = true ] ] ; then
2019-02-05 09:30:43 +00:00
if [ [ ! -d " ${ webroot } " ] ] ; then
2018-07-06 04:48:36 +00:00
# make the Web directory if necessary
2019-02-05 09:30:43 +00:00
install -d -m 0755 ${ webroot }
2018-07-06 04:48:36 +00:00
fi
if [ [ " ${ INSTALL_WEB_SERVER } " = = true ] ] ; then
# Set the owner and permissions
2019-02-05 09:30:43 +00:00
chown ${ LIGHTTPD_USER } :${ LIGHTTPD_GROUP } ${ webroot }
chmod 0775 ${ webroot }
2020-10-09 14:12:21 +00:00
# Repair permissions if webroot is not world readable
2019-05-05 20:34:13 +00:00
chmod a+rx /var/www
2020-10-09 14:12:21 +00:00
chmod a+rx ${ webroot }
2019-08-05 19:20:07 +00:00
# Give lighttpd access to the pihole group so the web interface can
# manage the gravity.db database
usermod -a -G pihole ${ LIGHTTPD_USER }
2018-07-06 04:48:36 +00:00
# If the lighttpd command is executable,
2018-11-01 19:00:08 +00:00
if is_command lighty-enable-mod ; then
2018-07-06 04:48:36 +00:00
# enable fastcgi and fastcgi-php
lighty-enable-mod fastcgi fastcgi-php > /dev/null || true
else
# Otherwise, show info about installing them
2018-10-31 18:14:05 +00:00
printf " %b Warning: 'lighty-enable-mod' utility not found\\n" " ${ INFO } "
printf " Please ensure fastcgi is enabled if you experience issues\\n"
2018-07-06 04:48:36 +00:00
fi
fi
fi
# Install base files and web interface
2018-07-29 16:06:51 +00:00
if ! installScripts; then
2018-10-31 18:14:05 +00:00
printf " %b Failure in dependent script copy function.\\n" " ${ CROSS } "
2018-07-29 16:06:51 +00:00
exit 1
fi
2018-07-06 04:48:36 +00:00
# Install config files
2018-07-30 21:02:18 +00:00
if ! installConfigs; then
2018-10-31 18:14:05 +00:00
printf " %b Failure in dependent config copy function.\\n" " ${ CROSS } "
2018-07-30 21:02:18 +00:00
exit 1
fi
2018-07-06 04:48:36 +00:00
# If the user wants to install the dashboard,
if [ [ " ${ INSTALL_WEB_INTERFACE } " = = true ] ] ; then
# do so
installPiholeWeb
fi
# Install the cron file
installCron
2021-07-28 19:18:02 +00:00
2018-07-06 04:48:36 +00:00
# Install the logrotate file
2021-07-28 19:18:02 +00:00
installLogrotate || true
2019-01-24 08:36:27 +00:00
# Check if dnsmasq is present. If so, disable it and back up any possible
# config file
disable_dnsmasq
2018-04-15 01:08:16 +00:00
2018-07-06 04:48:36 +00:00
# install a man page entry for pihole
install_manpage
# Update setupvars.conf with any variables that may or may not have been changed during the install
finalExports
2016-08-19 21:45:24 +00:00
}
2016-12-23 16:53:42 +00:00
2017-07-07 00:25:56 +00:00
# SELinux
2016-12-23 16:27:52 +00:00
checkSelinux( ) {
2019-10-14 18:14:45 +00:00
local DEFAULT_SELINUX
local CURRENT_SELINUX
local SELINUX_ENFORCING = 0
2020-05-12 21:45:56 +00:00
# Check for SELinux configuration file and getenforce command
2021-12-02 13:44:50 +00:00
if [ [ -f /etc/selinux/config ] ] && is_command getenforce; then
2020-05-12 21:45:56 +00:00
# Check the default SELinux mode
2019-10-14 18:14:45 +00:00
DEFAULT_SELINUX = $( awk -F= '/^SELINUX=/ {print $2}' /etc/selinux/config)
case " ${ DEFAULT_SELINUX ,, } " in
enforcing)
2019-11-14 19:06:23 +00:00
printf " %b %bDefault SELinux: %s%b\\n" " ${ CROSS } " " ${ COL_RED } " " ${ DEFAULT_SELINUX } " " ${ COL_NC } "
2019-10-14 18:14:45 +00:00
SELINUX_ENFORCING = 1
; ;
*) # 'permissive' and 'disabled'
2019-11-14 19:06:23 +00:00
printf " %b %bDefault SELinux: %s%b\\n" " ${ TICK } " " ${ COL_GREEN } " " ${ DEFAULT_SELINUX } " " ${ COL_NC } "
2019-10-14 18:14:45 +00:00
; ;
esac
# Check the current state of SELinux
CURRENT_SELINUX = $( getenforce)
case " ${ CURRENT_SELINUX ,, } " in
enforcing)
2019-11-14 19:06:23 +00:00
printf " %b %bCurrent SELinux: %s%b\\n" " ${ CROSS } " " ${ COL_RED } " " ${ CURRENT_SELINUX } " " ${ COL_NC } "
2019-10-14 18:14:45 +00:00
SELINUX_ENFORCING = 1
; ;
*) # 'permissive' and 'disabled'
2019-11-14 19:06:23 +00:00
printf " %b %bCurrent SELinux: %s%b\\n" " ${ TICK } " " ${ COL_GREEN } " " ${ CURRENT_SELINUX } " " ${ COL_NC } "
2019-10-14 18:14:45 +00:00
; ;
esac
else
2019-11-14 19:06:23 +00:00
echo -e " ${ INFO } ${ COL_GREEN } SELinux not detected ${ COL_NC } " ;
2019-10-14 18:14:45 +00:00
fi
# Exit the installer if any SELinux checks toggled the flag
if [ [ " ${ SELINUX_ENFORCING } " -eq 1 ] ] && [ [ -z " ${ PIHOLE_SELINUX } " ] ] ; then
2019-11-14 19:06:23 +00:00
printf " Pi-hole does not provide an SELinux policy as the required changes modify the security of your system.\\n"
printf " Please refer to https://wiki.centos.org/HowTos/SELinux if SELinux is required for your deployment.\\n"
2020-07-23 14:19:56 +00:00
printf " This check can be skipped by setting the environment variable %bPIHOLE_SELINUX%b to %btrue%b\\n" " ${ COL_LIGHT_RED } " " ${ COL_NC } " " ${ COL_LIGHT_RED } " " ${ COL_NC } "
printf " e.g: export PIHOLE_SELINUX=true\\n"
printf " By setting this variable to true you acknowledge there may be issues with Pi-hole during or after the install\\n"
2019-11-14 19:06:23 +00:00
printf "\\n %bSELinux Enforcing detected, exiting installer%b\\n" " ${ COL_LIGHT_RED } " " ${ COL_NC } " ;
2019-10-14 18:14:45 +00:00
exit 1;
2020-07-23 14:19:56 +00:00
elif [ [ " ${ SELINUX_ENFORCING } " -eq 1 ] ] && [ [ -n " ${ PIHOLE_SELINUX } " ] ] ; then
printf " %b %bSELinux Enforcing detected%b. PIHOLE_SELINUX env variable set - installer will continue\\n" " ${ INFO } " " ${ COL_LIGHT_RED } " " ${ COL_NC } "
2018-07-06 04:48:36 +00:00
fi
2015-11-07 18:07:50 +00:00
}
2017-07-07 00:25:56 +00:00
# Installation complete message with instructions for the user
2016-01-27 06:11:38 +00:00
displayFinalMessage( ) {
2021-03-18 01:35:27 +00:00
# If the number of arguments is > 0,
2018-07-06 04:48:36 +00:00
if [ [ " ${# 1 } " -gt 0 ] ] ; then
2021-03-18 01:35:27 +00:00
# set the password to the first argument.
2018-07-06 04:48:36 +00:00
pwstring = " $1 "
2021-10-05 15:40:12 +00:00
elif [ [ $( grep 'WEBPASSWORD' -c " ${ setupVars } " ) -gt 0 ] ] ; then
2021-03-18 01:35:27 +00:00
# Else if the password exists from previous setup, we'll load it later
2018-07-06 04:48:36 +00:00
pwstring = "unchanged"
else
2021-03-18 01:35:27 +00:00
# Else, inform the user that there is no set password.
2018-07-06 04:48:36 +00:00
pwstring = "NOT SET"
fi
# If the user wants to install the dashboard,
if [ [ " ${ INSTALL_WEB_INTERFACE } " = = true ] ] ; then
# Store a message in a variable and display it
2022-07-06 02:10:40 +00:00
additional = " View the web interface at http://pi.hole/admin or http:// ${ IPV4_ADDRESS %/* } /admin\\n\\nYour Admin Webpage login password is ${ pwstring } "
2021-11-25 06:41:40 +00:00
fi
2017-01-28 15:43:33 +00:00
2018-07-06 04:48:36 +00:00
# Final completion message to user
2022-07-04 21:00:16 +00:00
dialog --no-shadow --keep-tite \
2022-06-19 00:17:10 +00:00
--title "Installation Complete!" \
2022-07-02 19:36:16 +00:00
--msgbox " Configure your devices to use the Pi-hole as their DNS server using:\
\\ n\\ nIPv4: ${ IPV4_ADDRESS %/* } \
\\ nIPv6: ${ IPV6_ADDRESS :- "Not Configured" } \
\\ nIf you have not done so already, the above IP should be set to static.\
2022-07-06 02:10:40 +00:00
\\ n${ additional } " " ${ r } " " ${ c } "
2016-08-19 21:31:11 +00:00
}
2016-10-20 02:47:45 +00:00
update_dialogs( ) {
2018-07-06 04:48:36 +00:00
# If pihole -r "reconfigure" option was selected,
if [ [ " ${ reconfigure } " = true ] ] ; then
# set some variables that will be used
opt1a = "Repair"
opt1b = "This will retain existing settings"
strAdd = "You will remain on the same version"
else
2021-03-18 01:35:27 +00:00
# Otherwise, set some variables with different values
2018-07-06 04:48:36 +00:00
opt1a = "Update"
opt1b = "This will retain existing settings."
strAdd = "You will be updated to the latest version."
fi
opt2a = "Reconfigure"
Hardcode whiptail dimensions to 20 rows and 70 chars width
With the suggested way to call the installer via "curl -sSL https://install.pi-hole.net | bash", STDIN is no terminal, but overridden by the curl output, hence in most cases, the minimum dimensions were applied, even on larger screens. All whiptail calls are hence assured to work fine with those dimensions, aside of one case, making the calculations obsolete.
This commit hardcodes the whiptail dimensions to the prior minimum and removes the calculations. This also helps with testing, as it does not matter anymore how the script is called, and developers have a clearly defined space to make dialogs look nice, including line breaks, menu and list heights.
The only case which does not fit the 70 character width, the second menu entry of the "pihole -r" dialog, has been shortened accordingly. This was not an issue before, as "pihole -r" does not override the scripts STDIN and hence did allow larger dimensions based on the now removed calculations.
See the following discussions for reference:
- https://github.com/pi-hole/pi-hole/issues/3323
- https://github.com/pi-hole/pi-hole/pull/4197#issuecomment-876702380
Signed-off-by: MichaIng <micha@dietpi.com>
2021-07-16 17:58:49 +00:00
opt2b = "Resets Pi-hole and allows re-selecting settings."
2018-07-06 04:48:36 +00:00
# Display the information to the user
2022-07-04 21:00:16 +00:00
UpdateCmd = $( dialog --no-shadow --keep-tite --output-fd 1 \
2022-07-05 21:23:59 +00:00
--cancel-label Exit \
2022-06-19 00:17:10 +00:00
--title "Existing Install Detected!" \
2022-07-02 19:36:16 +00:00
--menu " \\n\\nWe have detected an existing install.\
\\ n\\ nPlease choose from the following options:\
\\ n( $strAdd ) " \
2022-06-19 00:17:10 +00:00
" ${ r } " " ${ c } " 2 \
2018-07-06 04:48:36 +00:00
" ${ opt1a } " " ${ opt1b } " \
2022-07-04 21:00:16 +00:00
" ${ opt2a } " " ${ opt2b } " || true )
2022-06-19 00:17:10 +00:00
result = $?
case ${ result } in
" ${ DIALOG_CANCEL } " | " ${ DIALOG_ESC } " )
2022-07-06 10:29:03 +00:00
printf " %b Cancel was selected, exiting installer%b\\n" " ${ COL_LIGHT_RED } " " ${ COL_NC } "
2022-06-19 00:17:10 +00:00
exit 1
; ;
esac
2018-07-06 04:48:36 +00:00
# Set the variable based on if the user chooses
case ${ UpdateCmd } in
# repair, or
2022-06-19 00:17:10 +00:00
" ${ opt1a } " )
2018-10-31 18:14:05 +00:00
printf " %b %s option selected\\n" " ${ INFO } " " ${ opt1a } "
2018-07-06 04:48:36 +00:00
useUpdateVars = true
; ;
# reconfigure,
2022-06-19 00:17:10 +00:00
" ${ opt2a } " )
2018-10-31 18:14:05 +00:00
printf " %b %s option selected\\n" " ${ INFO } " " ${ opt2a } "
2018-07-06 04:48:36 +00:00
useUpdateVars = false
; ;
2016-12-21 01:22:57 +00:00
esac
2016-01-01 02:26:05 +00:00
}
2018-05-01 20:59:12 +00:00
check_download_exists( ) {
2018-07-06 04:48:36 +00:00
status = $( curl --head --silent " https://ftl.pi-hole.net/ ${ 1 } " | head -n 1)
if grep -q "404" <<< " $status " ; then
return 1
else
return 0
fi
2018-05-01 20:59:12 +00:00
}
fully_fetch_repo( ) {
2018-07-06 04:48:36 +00:00
# Add upstream branches to shallow clone
local directory = " ${ 1 } "
2018-05-01 20:59:12 +00:00
2018-07-06 04:48:36 +00:00
cd " ${ directory } " || return 1
if is_repo " ${ directory } " ; then
git remote set-branches origin '*' || return 1
git fetch --quiet || return 1
else
return 1
fi
return 0
2018-05-01 20:59:12 +00:00
}
get_available_branches( ) {
2018-07-06 04:48:36 +00:00
# Return available branches
local directory
directory = " ${ 1 } "
local output
cd " ${ directory } " || return 1
# Get reachable remote branches, but store STDERR as STDOUT variable
2018-07-28 03:20:59 +00:00
output = $( { git ls-remote --heads --quiet | cut -d'/' -f3- -; } 2>& 1 )
2018-10-31 18:14:05 +00:00
# echo status for calling function to capture
2018-07-06 04:48:36 +00:00
echo " $output "
return
2018-05-01 20:59:12 +00:00
}
fetch_checkout_pull_branch( ) {
2018-07-06 04:48:36 +00:00
# Check out specified branch
local directory
directory = " ${ 1 } "
local branch
branch = " ${ 2 } "
# Set the reference for the requested branch, fetch, check it put and pull it
cd " ${ directory } " || return 1
git remote set-branches origin " ${ branch } " || return 1
git stash --all --quiet & > /dev/null || true
git clean --quiet --force -d || true
git fetch --quiet || return 1
checkout_pull_branch " ${ directory } " " ${ branch } " || return 1
2018-05-01 20:59:12 +00:00
}
checkout_pull_branch( ) {
2018-07-06 04:48:36 +00:00
# Check out specified branch
local directory
directory = " ${ 1 } "
local branch
branch = " ${ 2 } "
local oldbranch
2018-05-01 20:59:12 +00:00
2018-07-06 04:48:36 +00:00
cd " ${ directory } " || return 1
2018-05-01 20:59:12 +00:00
2018-07-06 04:48:36 +00:00
oldbranch = " $( git symbolic-ref HEAD) "
2018-05-01 20:59:12 +00:00
2018-07-06 04:48:36 +00:00
str = " Switching to branch: ' ${ branch } ' from ' ${ oldbranch } ' "
2018-10-31 18:14:05 +00:00
printf " %b %s" " ${ INFO } " " $str "
2018-07-06 04:48:36 +00:00
git checkout " ${ branch } " --quiet || return 1
2018-10-31 18:14:05 +00:00
printf "%b %b %s\\n" " ${ OVER } " " ${ TICK } " " $str "
2019-05-03 10:27:56 +00:00
# Data in the repositories is public anyway so we can make it readable by everyone (+r to keep executable permission if already set by git)
2019-05-07 08:46:16 +00:00
chmod -R a+rX " ${ directory } "
2018-05-01 20:59:12 +00:00
2021-10-06 00:19:28 +00:00
git_pull = $( git pull --no-rebase || return 1)
2018-05-01 20:59:12 +00:00
2018-07-06 04:48:36 +00:00
if [ [ " $git_pull " = = *"up-to-date" * ] ] ; then
2018-10-31 18:14:05 +00:00
printf " %b %s\\n" " ${ INFO } " " ${ git_pull } "
2018-07-06 04:48:36 +00:00
else
2018-10-31 18:14:05 +00:00
printf "%s\\n" " $git_pull "
2018-07-06 04:48:36 +00:00
fi
2018-05-01 20:59:12 +00:00
2018-07-06 04:48:36 +00:00
return 0
2018-05-01 20:59:12 +00:00
}
2017-01-28 15:15:42 +00:00
clone_or_update_repos( ) {
2018-07-06 04:48:36 +00:00
# If the user wants to reconfigure,
if [ [ " ${ reconfigure } " = = true ] ] ; then
2018-10-31 18:14:05 +00:00
printf " %b Performing reconfiguration, skipping download of local repos\\n" " ${ INFO } "
2018-07-06 04:48:36 +00:00
# Reset the Core repo
resetRepo ${ PI_HOLE_LOCAL_REPO } || \
2022-07-06 10:29:03 +00:00
{ printf " %b Unable to reset %s, exiting installer%b\\n" " ${ COL_LIGHT_RED } " " ${ PI_HOLE_LOCAL_REPO } " " ${ COL_NC } " ; \
2017-05-22 21:43:52 +00:00
exit 1; \
2017-01-28 15:15:42 +00:00
}
2018-07-06 04:48:36 +00:00
# If the Web interface was installed,
if [ [ " ${ INSTALL_WEB_INTERFACE } " = = true ] ] ; then
# reset it's repo
resetRepo ${ webInterfaceDir } || \
2022-07-06 10:29:03 +00:00
{ printf " %b Unable to reset %s, exiting installer%b\\n" " ${ COL_LIGHT_RED } " " ${ webInterfaceDir } " " ${ COL_NC } " ; \
2018-07-06 04:48:36 +00:00
exit 1; \
}
fi
# Otherwise, a repair is happening
else
# so get git files for Core
getGitFiles ${ PI_HOLE_LOCAL_REPO } ${ piholeGitUrl } || \
2022-07-06 10:29:03 +00:00
{ printf " %b Unable to clone %s into %s, unable to continue%b\\n" " ${ COL_LIGHT_RED } " " ${ piholeGitUrl } " " ${ PI_HOLE_LOCAL_REPO } " " ${ COL_NC } " ; \
2017-05-21 20:47:26 +00:00
exit 1; \
2017-01-28 15:15:42 +00:00
}
2018-07-06 04:48:36 +00:00
# If the Web interface was installed,
if [ [ " ${ INSTALL_WEB_INTERFACE } " = = true ] ] ; then
# get the Web git files
getGitFiles ${ webInterfaceDir } ${ webInterfaceGitUrl } || \
2022-07-06 10:29:03 +00:00
{ printf " %b Unable to clone %s into ${ webInterfaceDir } , exiting installer%b\\n " " ${ COL_LIGHT_RED } " " ${ webInterfaceGitUrl } " " ${ COL_NC } " ; \
2018-07-06 04:48:36 +00:00
exit 1; \
}
fi
fi
2017-01-28 15:15:42 +00:00
}
2018-02-23 04:13:38 +00:00
# Download FTL binary to random temp directory and install FTL binary
2019-11-14 18:52:07 +00:00
# Disable directive for SC2120 a value _can_ be passed to this function, but it is passed from an external script that sources this one
# shellcheck disable=SC2120
2017-02-20 16:24:19 +00:00
FTLinstall( ) {
2018-07-06 04:48:36 +00:00
# Local, named variables
local str = "Downloading and Installing FTL"
2018-10-31 18:14:05 +00:00
printf " %b %s..." " ${ INFO } " " ${ str } "
2018-07-06 04:48:36 +00:00
# Move into the temp ftl directory
2018-10-31 18:14:05 +00:00
pushd " $( mktemp -d) " > /dev/null || { printf "Unable to make temporary directory for FTL binary download\\n" ; return 1; }
2018-03-18 00:38:34 +00:00
2018-07-06 04:48:36 +00:00
local ftlBranch
local url
if [ [ -f "/etc/pihole/ftlbranch" ] ] ; then
ftlBranch = $( </etc/pihole/ftlbranch)
else
ftlBranch = "master"
fi
2019-11-14 18:52:07 +00:00
local binary
binary = " ${ 1 } "
2018-07-06 04:48:36 +00:00
# Determine which version of FTL to download
if [ [ " ${ ftlBranch } " = = "master" ] ] ; then
2020-02-17 21:29:25 +00:00
url = "https://github.com/pi-hole/ftl/releases/latest/download"
2018-07-06 04:48:36 +00:00
else
url = " https://ftl.pi-hole.net/ ${ ftlBranch } "
fi
if curl -sSL --fail " ${ url } / ${ binary } " -o " ${ binary } " ; then
2021-03-18 01:35:27 +00:00
# If the download worked, get sha1 of the binary we just downloaded for verification.
2018-07-06 04:48:36 +00:00
curl -sSL --fail " ${ url } / ${ binary } .sha1 " -o " ${ binary } .sha1 "
# If we downloaded binary file (as opposed to text),
if sha1sum --status --quiet -c " ${ binary } " .sha1; then
2018-10-31 18:14:05 +00:00
printf "transferred... "
2019-01-09 18:26:19 +00:00
# Before stopping FTL, we download the macvendor database
2019-01-28 19:48:30 +00:00
curl -sSL "https://ftl.pi-hole.net/macvendor.db" -o " ${ PI_HOLE_CONFIG_DIR } /macvendor.db " || true
2019-01-09 18:26:19 +00:00
2019-02-01 16:39:12 +00:00
# Stop pihole-FTL service if available
2018-07-06 04:48:36 +00:00
stop_service pihole-FTL & > /dev/null
2019-02-01 16:39:12 +00:00
2018-07-06 04:48:36 +00:00
# Install the new version with the correct permissions
install -T -m 0755 " ${ binary } " /usr/bin/pihole-FTL
2019-02-01 16:39:12 +00:00
2018-07-06 04:48:36 +00:00
# Move back into the original directory the user was in
2018-10-31 18:14:05 +00:00
popd > /dev/null || { printf "Unable to return to original directory after FTL binary download.\\n" ; return 1; }
2018-07-06 04:48:36 +00:00
2019-02-01 16:39:12 +00:00
# Installed the FTL service
2018-10-31 18:14:05 +00:00
printf "%b %b %s\\n" " ${ OVER } " " ${ TICK } " " ${ str } "
2018-07-06 04:48:36 +00:00
return 0
else
2021-03-18 01:35:27 +00:00
# Otherwise, the hash download failed, so print and exit.
2018-10-31 18:14:05 +00:00
popd > /dev/null || { printf "Unable to return to original directory after FTL binary download.\\n" ; return 1; }
printf "%b %b %s\\n" " ${ OVER } " " ${ CROSS } " " ${ str } "
2022-07-06 10:29:03 +00:00
printf " %b Error: Download of %s/%s failed (checksum error)%b\\n" " ${ COL_LIGHT_RED } " " ${ url } " " ${ binary } " " ${ COL_NC } "
2018-07-06 04:48:36 +00:00
return 1
fi
2017-02-20 17:36:24 +00:00
else
2021-03-18 01:35:27 +00:00
# Otherwise, the download failed, so print and exit.
2018-10-31 18:14:05 +00:00
popd > /dev/null || { printf "Unable to return to original directory after FTL binary download.\\n" ; return 1; }
printf "%b %b %s\\n" " ${ OVER } " " ${ CROSS } " " ${ str } "
2018-07-06 04:48:36 +00:00
# The URL could not be found
2022-07-06 10:29:03 +00:00
printf " %b Error: URL %s/%s not found%b\\n" " ${ COL_LIGHT_RED } " " ${ url } " " ${ binary } " " ${ COL_NC } "
2018-07-06 04:48:36 +00:00
return 1
fi
2017-02-20 16:24:19 +00:00
}
2019-01-24 08:31:02 +00:00
disable_dnsmasq( ) {
# dnsmasq can now be stopped and disabled if it exists
2021-12-02 13:44:50 +00:00
if is_command dnsmasq; then
2019-01-24 08:31:02 +00:00
if check_service_active "dnsmasq" ; then
printf " %b FTL can now resolve DNS Queries without dnsmasq running separately\\n" " ${ INFO } "
stop_service dnsmasq
disable_service dnsmasq
fi
fi
# Backup existing /etc/dnsmasq.conf if present and ensure that
# /etc/dnsmasq.conf contains only "conf-dir=/etc/dnsmasq.d"
local conffile = "/etc/dnsmasq.conf"
if [ [ -f " ${ conffile } " ] ] ; then
printf " %b Backing up %s to %s.old\\n" " ${ INFO } " " ${ conffile } " " ${ conffile } "
mv " ${ conffile } " " ${ conffile } .old "
fi
# Create /etc/dnsmasq.conf
echo "conf-dir=/etc/dnsmasq.d" > " ${ conffile } "
2019-05-04 22:32:10 +00:00
chmod 644 " ${ conffile } "
2019-01-24 08:31:02 +00:00
}
2018-04-02 20:53:32 +00:00
get_binary_name( ) {
2018-07-06 04:48:36 +00:00
# This gives the machine architecture which may be different from the OS architecture...
local machine
machine = $( uname -m)
2019-11-14 18:52:07 +00:00
local l_binary
2020-10-13 13:18:14 +00:00
local str = "Detecting processor"
2018-10-31 18:14:05 +00:00
printf " %b %s..." " ${ INFO } " " ${ str } "
2018-07-06 04:48:36 +00:00
# If the machine is arm or aarch
if [ [ " ${ machine } " = = "arm" * || " ${ machine } " = = *"aarch" * ] ] ; then
# ARM
local rev
rev = $( uname -m | sed "s/[^0-9]//g;" )
local lib
2022-04-20 17:21:24 +00:00
lib = $( ldd " $( which sh) " | grep -E '^\s*/lib' | awk '{ print $1 }' )
2018-07-06 04:48:36 +00:00
if [ [ " ${ lib } " = = "/lib/ld-linux-aarch64.so.1" ] ] ; then
2020-10-13 13:18:14 +00:00
printf "%b %b Detected AArch64 (64 Bit ARM) processor\\n" " ${ OVER } " " ${ TICK } "
2018-07-06 04:48:36 +00:00
# set the binary to be used
2019-11-14 18:52:07 +00:00
l_binary = "pihole-FTL-aarch64-linux-gnu"
2018-07-06 04:48:36 +00:00
elif [ [ " ${ lib } " = = "/lib/ld-linux-armhf.so.3" ] ] ; then
2020-10-13 13:18:14 +00:00
# Hard-float available: Use gnueabihf binaries
# If ARMv8 or higher is found (e.g., BCM2837 as found in Raspberry Pi Model 3B)
if [ [ " ${ rev } " -gt 7 ] ] ; then
printf "%b %b Detected ARMv8 (or newer) processor\\n" " ${ OVER } " " ${ TICK } "
2018-07-06 04:48:36 +00:00
# set the binary to be used
2020-10-13 13:18:14 +00:00
l_binary = "pihole-FTL-armv8-linux-gnueabihf"
elif [ [ " ${ rev } " -eq 7 ] ] ; then
2021-03-18 01:35:27 +00:00
# Otherwise, if ARMv7 is found (e.g., BCM2836 as found in Raspberry Pi Model 2)
2020-10-13 13:18:14 +00:00
printf "%b %b Detected ARMv7 processor (with hard-float support)\\n" " ${ OVER } " " ${ TICK } "
# set the binary to be used
l_binary = "pihole-FTL-armv7-linux-gnueabihf"
2018-07-06 04:48:36 +00:00
else
2021-03-18 01:35:27 +00:00
# Otherwise, use the ARMv6 binary (e.g., BCM2835 as found in Raspberry Pi Zero and Model 1)
2020-10-13 13:18:14 +00:00
printf "%b %b Detected ARMv6 processor (with hard-float support)\\n" " ${ OVER } " " ${ TICK } "
2018-07-06 04:48:36 +00:00
# set the binary to be used
2020-10-13 13:18:14 +00:00
l_binary = "pihole-FTL-armv6-linux-gnueabihf"
2018-07-06 04:48:36 +00:00
fi
else
2020-10-13 13:18:14 +00:00
# No hard-float support found: Use gnueabi binaries
# Use the ARMv4-compliant binary only if we detected an ARMv4T core
if [ [ " ${ rev } " -eq 4 ] ] ; then
printf "%b %b Detected ARMv4 processor\\n" " ${ OVER } " " ${ TICK } "
2019-10-30 03:26:46 +00:00
# set the binary to be used
2020-10-13 13:18:14 +00:00
l_binary = "pihole-FTL-armv4-linux-gnueabi"
# Otherwise, use the ARMv5 binary. To date (end of 2020), all modern ARM processors
# are backwards-compatible to the ARMv5
2019-10-30 03:26:46 +00:00
else
2020-10-13 13:18:14 +00:00
printf "%b %b Detected ARMv5 (or newer) processor\\n" " ${ OVER } " " ${ TICK } "
2019-10-30 03:26:46 +00:00
# set the binary to be used
2020-10-13 13:18:14 +00:00
l_binary = "pihole-FTL-armv5-linux-gnueabi"
2019-10-30 03:26:46 +00:00
fi
2018-07-06 04:48:36 +00:00
fi
elif [ [ " ${ machine } " = = "x86_64" ] ] ; then
2020-10-13 13:18:14 +00:00
# This gives the processor of packages dpkg installs (for example, "i386")
2018-07-06 04:48:36 +00:00
local dpkgarch
2020-12-23 22:59:28 +00:00
dpkgarch = $( dpkg --print-processor 2> /dev/null || dpkg --print-architecture 2> /dev/null)
2018-07-06 04:48:36 +00:00
# Special case: This is a 32 bit OS, installed on a 64 bit machine
2020-10-13 13:18:14 +00:00
# -> change machine processor to download the 32 bit executable
2019-02-22 17:32:36 +00:00
# We only check this for Debian-based systems as this has been an issue
# in the past (see https://github.com/pi-hole/pi-hole/pull/2004)
2018-07-06 04:48:36 +00:00
if [ [ " ${ dpkgarch } " = = "i386" ] ] ; then
2020-10-13 13:18:14 +00:00
printf "%b %b Detected 32bit (i686) processor\\n" " ${ OVER } " " ${ TICK } "
2019-11-14 18:52:07 +00:00
l_binary = "pihole-FTL-linux-x86_32"
2018-07-06 04:48:36 +00:00
else
# 64bit
2020-10-13 13:18:14 +00:00
printf "%b %b Detected x86_64 processor\\n" " ${ OVER } " " ${ TICK } "
2018-07-06 04:48:36 +00:00
# set the binary to be used
2019-11-14 18:52:07 +00:00
l_binary = "pihole-FTL-linux-x86_64"
2018-07-06 04:48:36 +00:00
fi
2017-02-13 15:50:48 +00:00
else
2018-07-06 04:48:36 +00:00
# Something else - we try to use 32bit executable and warn the user
if [ [ ! " ${ machine } " = = "i686" ] ] ; then
2018-10-31 18:14:05 +00:00
printf "%b %b %s...\\n" " ${ OVER } " " ${ CROSS } " " ${ str } "
2020-10-13 13:18:14 +00:00
printf " %b %bNot able to detect processor (unknown: %s), trying x86 (32bit) executable%b\\n" " ${ INFO } " " ${ COL_LIGHT_RED } " " ${ machine } " " ${ COL_NC } "
2018-10-31 18:14:05 +00:00
printf " %b Contact Pi-hole Support if you experience issues (e.g: FTL not running)\\n" " ${ INFO } "
2018-07-06 04:48:36 +00:00
else
2020-10-13 13:18:14 +00:00
printf "%b %b Detected 32bit (i686) processor\\n" " ${ OVER } " " ${ TICK } "
2018-07-06 04:48:36 +00:00
fi
2019-11-14 18:52:07 +00:00
l_binary = "pihole-FTL-linux-x86_32"
2017-02-13 13:42:11 +00:00
fi
2019-11-14 18:52:07 +00:00
2021-03-18 01:35:27 +00:00
# Returning a string value via echo
2019-11-14 18:52:07 +00:00
echo ${ l_binary }
2018-04-02 20:53:32 +00:00
}
2017-02-13 09:29:27 +00:00
2018-07-06 04:48:36 +00:00
FTLcheckUpdate( ) {
#In the next section we check to see if FTL is already installed (in case of pihole -r).
#If the installed version matches the latest version, then check the installed sha1sum of the binary vs the remote sha1sum. If they do not match, then download
2018-10-31 18:14:05 +00:00
printf " %b Checking for existing FTL binary...\\n" " ${ INFO } "
2017-07-12 21:02:07 +00:00
2018-07-06 04:48:36 +00:00
local ftlLoc
2022-01-01 18:02:20 +00:00
ftlLoc = $( command -v pihole-FTL 2>/dev/null)
2018-03-18 00:38:34 +00:00
2018-07-06 04:48:36 +00:00
local ftlBranch
2018-03-18 00:38:34 +00:00
2018-07-06 04:48:36 +00:00
if [ [ -f "/etc/pihole/ftlbranch" ] ] ; then
ftlBranch = $( </etc/pihole/ftlbranch)
else
ftlBranch = "master"
2018-05-01 20:59:12 +00:00
fi
2019-11-14 18:52:07 +00:00
local binary
binary = " ${ 1 } "
2018-07-06 04:48:36 +00:00
local remoteSha1
local localSha1
2018-04-02 20:53:32 +00:00
2018-07-06 04:48:36 +00:00
# if dnsmasq exists and is running at this point, force reinstall of FTL Binary
2021-12-02 13:44:50 +00:00
if is_command dnsmasq; then
2018-07-06 04:48:36 +00:00
if check_service_active "dnsmasq" ; then
return 0
fi
2018-04-02 20:53:32 +00:00
fi
2017-07-24 22:22:04 +00:00
2018-07-06 04:48:36 +00:00
if [ [ ! " ${ ftlBranch } " = = "master" ] ] ; then
#Check whether or not the binary for this FTL branch actually exists. If not, then there is no update!
local path
path = " ${ ftlBranch } / ${ binary } "
# shellcheck disable=SC1090
if ! check_download_exists " $path " ; then
2018-10-31 18:14:05 +00:00
printf " %b Branch \"%s\" is not available.\\n" " ${ INFO } " " ${ ftlBranch } "
printf " %b Use %bpihole checkout ftl [branchname]%b to switch to a valid branch.\\n" " ${ INFO } " " ${ COL_LIGHT_GREEN } " " ${ COL_NC } "
2018-07-06 04:48:36 +00:00
return 2
fi
2017-07-12 21:02:07 +00:00
2018-07-06 04:48:36 +00:00
if [ [ ${ ftlLoc } ] ] ; then
# We already have a pihole-FTL binary downloaded.
# Alt branches don't have a tagged version against them, so just confirm the checksum of the local vs remote to decide whether we download or not
remoteSha1 = $( curl -sSL --fail " https://ftl.pi-hole.net/ ${ ftlBranch } / ${ binary } .sha1 " | cut -d ' ' -f 1)
2022-01-01 18:02:20 +00:00
localSha1 = $( sha1sum " $( command -v pihole-FTL) " | cut -d ' ' -f 1)
2018-07-06 04:48:36 +00:00
if [ [ " ${ remoteSha1 } " != " ${ localSha1 } " ] ] ; then
2018-10-31 18:14:05 +00:00
printf " %b Checksums do not match, downloading from ftl.pi-hole.net.\\n" " ${ INFO } "
2018-07-06 04:48:36 +00:00
return 0
else
2018-10-31 18:14:05 +00:00
printf " %b Checksum of installed binary matches remote. No need to download!\\n" " ${ INFO } "
2018-07-06 04:48:36 +00:00
return 1
fi
2018-03-06 18:44:57 +00:00
else
2018-07-06 04:48:36 +00:00
return 0
2018-03-06 18:44:57 +00:00
fi
else
2018-07-06 04:48:36 +00:00
if [ [ ${ ftlLoc } ] ] ; then
local FTLversion
FTLversion = $( /usr/bin/pihole-FTL tag)
local FTLlatesttag
2021-01-12 21:42:51 +00:00
if ! FTLlatesttag = $( curl -sI https://github.com/pi-hole/FTL/releases/latest | grep --color= never -i Location: | awk -F / '{print $NF}' | tr -d '[:cntrl:]' ) ; then
2019-06-29 03:49:56 +00:00
# There was an issue while retrieving the latest version
2019-07-20 00:35:21 +00:00
printf " %b Failed to retrieve latest FTL release metadata" " ${ CROSS } "
2019-06-29 03:49:56 +00:00
return 3
fi
2018-07-06 04:48:36 +00:00
if [ [ " ${ FTLversion } " != " ${ FTLlatesttag } " ] ] ; then
return 0
else
2018-10-31 18:14:05 +00:00
printf " %b Latest FTL Binary already installed (%s). Confirming Checksum...\\n" " ${ INFO } " " ${ FTLlatesttag } "
2018-07-06 04:48:36 +00:00
remoteSha1 = $( curl -sSL --fail " https://github.com/pi-hole/FTL/releases/download/ ${ FTLversion % $'\r' } / ${ binary } .sha1 " | cut -d ' ' -f 1)
2022-01-01 18:02:20 +00:00
localSha1 = $( sha1sum " $( command -v pihole-FTL) " | cut -d ' ' -f 1)
2018-07-06 04:48:36 +00:00
if [ [ " ${ remoteSha1 } " != " ${ localSha1 } " ] ] ; then
2018-10-31 18:14:05 +00:00
printf " %b Corruption detected...\\n" " ${ INFO } "
2018-07-06 04:48:36 +00:00
return 0
else
2018-10-31 18:14:05 +00:00
printf " %b Checksum correct. No need to download!\\n" " ${ INFO } "
2018-07-06 04:48:36 +00:00
return 1
fi
fi
else
return 0
fi
2018-03-06 18:44:57 +00:00
fi
2018-04-02 20:53:32 +00:00
}
# Detect suitable FTL binary platform
FTLdetect( ) {
2018-10-31 18:14:05 +00:00
printf "\\n %b FTL Checks...\\n\\n" " ${ INFO } "
2018-04-02 20:53:32 +00:00
2019-11-14 18:52:07 +00:00
printf " %b" " ${ 2 } "
if FTLcheckUpdate " ${ 1 } " ; then
FTLinstall " ${ 1 } " || return 1
2018-07-06 04:48:36 +00:00
fi
2018-01-20 13:55:48 +00:00
}
2017-07-25 21:49:06 +00:00
2018-01-20 13:55:48 +00:00
make_temporary_log( ) {
2018-07-06 04:48:36 +00:00
# Create a random temporary file for the log
TEMPLOG = $( mktemp /tmp/pihole_temp.XXXXXX)
# Open handle 3 for templog
# https://stackoverflow.com/questions/18460186/writing-outputs-to-log-file-and-console
exec 3>" $TEMPLOG "
# Delete templog, but allow for addressing via file handle
# This lets us write to the log without having a temporary file on the drive, which
# is meant to be a security measure so there is not a lingering file on the drive during the install process
rm " $TEMPLOG "
2018-01-20 13:55:48 +00:00
}
2017-02-13 13:47:06 +00:00
2018-01-20 13:55:48 +00:00
copy_to_install_log( ) {
2018-07-06 04:48:36 +00:00
# Copy the contents of file descriptor 3 into the install log
# Since we use color codes such as '\e[1;33m', they should be removed
sed 's/ \[[0-9;]\{1,5\}m//g' < /proc/$$ /fd/3 > " ${ installLogLoc } "
2019-05-01 09:20:26 +00:00
chmod 644 " ${ installLogLoc } "
2017-02-13 10:07:29 +00:00
}
2017-02-13 09:29:27 +00:00
2016-10-08 19:17:04 +00:00
main( ) {
2018-07-06 04:48:36 +00:00
######## FIRST CHECK ########
# Must be root to install
local str = "Root user check"
2018-10-31 18:14:05 +00:00
printf "\\n"
2018-07-06 04:48:36 +00:00
# If the user's id is zero,
if [ [ " ${ EUID } " -eq 0 ] ] ; then
# they are root and all is good
2018-10-31 18:14:05 +00:00
printf " %b %s\\n" " ${ TICK } " " ${ str } "
2018-07-06 04:48:36 +00:00
# Show the Pi-hole logo so people know it's genuine since the logo and name are trademarked
show_ascii_berry
make_temporary_log
2017-01-01 14:45:03 +00:00
else
2021-03-18 01:35:27 +00:00
# Otherwise, they do not have enough privileges, so let the user know
2020-10-29 19:37:08 +00:00
printf " %b %s\\n" " ${ INFO } " " ${ str } "
2018-10-31 18:14:05 +00:00
printf " %b %bScript called with non-root privileges%b\\n" " ${ INFO } " " ${ COL_LIGHT_RED } " " ${ COL_NC } "
printf " The Pi-hole requires elevated privileges to install and run\\n"
printf " Please check the installer for any concerns regarding this requirement\\n"
printf " Make sure to download this script from a trusted source\\n\\n"
printf " %b Sudo utility check" " ${ INFO } "
2018-07-06 04:48:36 +00:00
2021-03-18 01:35:27 +00:00
# If the sudo command exists, try rerunning as admin
2018-11-01 19:00:08 +00:00
if is_command sudo ; then
2018-10-31 18:14:05 +00:00
printf "%b %b Sudo utility check\\n" " ${ OVER } " " ${ TICK } "
2020-10-30 15:53:20 +00:00
# when run via curl piping
if [ [ " $0 " = = "bash" ] ] ; then
# Download the install script and run it with admin rights
exec curl -sSL https://raw.githubusercontent.com/pi-hole/pi-hole/master/automated%20install/basic-install.sh | sudo bash " $@ "
else
# when run via calling local bash script
exec sudo bash " $0 " " $@ "
fi
2018-07-06 04:48:36 +00:00
exit $?
else
2021-03-18 01:35:27 +00:00
# Otherwise, tell the user they need to run the script as root, and bail
2018-10-31 18:14:05 +00:00
printf "%b %b Sudo utility check\\n" " ${ OVER } " " ${ CROSS } "
printf " %b Sudo is needed for the Web Interface to run pihole commands\\n\\n" " ${ INFO } "
printf " %b %bPlease re-run this installer as root ${ COL_NC } \\n " " ${ INFO } " " ${ COL_LIGHT_RED } "
2018-07-06 04:48:36 +00:00
exit 1
fi
fi
2022-07-04 21:00:16 +00:00
# Check if SELinux is Enforcing and exit before doing anything else
checkSelinux
2021-08-03 21:53:29 +00:00
# Check for supported package managers so that we may install dependencies
2021-08-03 20:48:26 +00:00
package_manager_detect
2018-07-06 04:48:36 +00:00
# Notify user of package availability
notify_package_updates_available
2021-08-03 21:48:03 +00:00
# Install packages necessary to perform os_check
2021-08-03 22:26:59 +00:00
printf " %b Checking for / installing Required dependencies for OS Check...\\n" " ${ INFO } "
2021-08-03 21:48:03 +00:00
install_dependent_packages " ${ OS_CHECK_DEPS [@] } "
2018-07-06 04:48:36 +00:00
2020-06-13 20:33:03 +00:00
# Check that the installed OS is officially supported - display warning if not
os_check
2021-08-03 21:48:03 +00:00
# Install packages used by this installation script
2021-08-03 22:26:59 +00:00
printf " %b Checking for / installing Required dependencies for this install script...\\n" " ${ INFO } "
2021-08-03 21:48:03 +00:00
install_dependent_packages " ${ INSTALLER_DEPS [@] } "
2021-09-12 20:40:37 +00:00
#In case of RPM based distro, select the proper PHP version
if [ [ " $PKG_MANAGER " = = "yum" || " $PKG_MANAGER " = = "dnf" ] ] ; then
2021-11-25 06:41:40 +00:00
select_rpm_php
2021-09-12 20:40:37 +00:00
fi
2018-07-06 04:48:36 +00:00
2021-08-03 21:53:29 +00:00
# If the setup variable file exists,
if [ [ -f " ${ setupVars } " ] ] ; then
# if it's running unattended,
if [ [ " ${ runUnattended } " = = true ] ] ; then
2022-06-19 00:17:10 +00:00
printf " %b Performing unattended setup, no dialogs will be displayed\\n" " ${ INFO } "
2021-08-03 21:53:29 +00:00
# Use the setup variables
useUpdateVars = true
# also disable debconf-apt-progress dialogs
export DEBIAN_FRONTEND = "noninteractive"
else
# If running attended, show the available options (repair/reconfigure)
update_dialogs
fi
fi
2018-07-06 04:48:36 +00:00
if [ [ " ${ useUpdateVars } " = = false ] ] ; then
# Display welcome dialogs
welcomeDialogs
# Create directory for Pi-hole storage
2019-05-03 10:27:56 +00:00
install -d -m 755 /etc/pihole/
2018-07-06 04:48:36 +00:00
# Determine available interfaces
get_available_interfaces
# Find interfaces and let the user choose one
chooseInterface
2021-11-06 20:32:03 +00:00
# find IPv4 and IPv6 information of the device
collect_v4andv6_information
2018-07-06 04:48:36 +00:00
# Decide what upstream DNS Servers to use
setDNS
# Give the user a choice of blocklists to include in their install. Or not.
chooseBlocklists
# Let the user decide if they want the web interface to be installed automatically
setAdminFlag
# Let the user decide if they want query logging enabled...
setLogging
2018-08-20 23:04:58 +00:00
# Let the user decide the FTL privacy level
setPrivacyLevel
2017-07-26 13:34:40 +00:00
else
2018-08-12 21:19:16 +00:00
# Setup adlist file if not exists
installDefaultBlocklists
2018-08-20 23:04:58 +00:00
2018-07-06 04:48:36 +00:00
# Source ${setupVars} to use predefined user variables in the functions
2019-05-10 14:13:23 +00:00
source " ${ setupVars } "
2018-08-20 23:04:58 +00:00
# Get the privacy level if it exists (default is 0)
if [ [ -f " ${ PI_HOLE_CONFIG_DIR } /pihole-FTL.conf " ] ] ; then
PRIVACY_LEVEL = $( sed -ne 's/PRIVACYLEVEL=\(.*\)/\1/p' " ${ PI_HOLE_CONFIG_DIR } /pihole-FTL.conf " )
# If no setting was found, default to 0
PRIVACY_LEVEL = " ${ PRIVACY_LEVEL :- 0 } "
fi
2018-07-06 04:48:36 +00:00
fi
2021-03-18 01:35:27 +00:00
# Download or update the scripts by updating the appropriate git repos
2018-07-06 04:48:36 +00:00
clone_or_update_repos
# Install the Core dependencies
local dep_install_list = ( " ${ PIHOLE_DEPS [@] } " )
if [ [ " ${ INSTALL_WEB_SERVER } " = = true ] ] ; then
2021-03-18 01:35:27 +00:00
# And, if the setting says so, install the Web admin interface dependencies
2018-07-06 04:48:36 +00:00
dep_install_list += ( " ${ PIHOLE_WEB_DEPS [@] } " )
fi
2021-08-03 22:26:59 +00:00
# Install packages used by the actual software
printf " %b Checking for / installing Required dependencies for Pi-hole software...\\n" " ${ INFO } "
2019-05-10 14:37:38 +00:00
install_dependent_packages " ${ dep_install_list [@] } "
2018-07-06 04:48:36 +00:00
unset dep_install_list
# On some systems, lighttpd is not enabled on first install. We need to enable it here if the user
2021-03-18 01:35:27 +00:00
# has chosen to install the web interface, else the LIGHTTPD_ENABLED check will fail
2018-07-06 04:48:36 +00:00
if [ [ " ${ INSTALL_WEB_SERVER } " = = true ] ] ; then
enable_service lighttpd
fi
# Determine if lighttpd is correctly enabled
if check_service_active "lighttpd" ; then
LIGHTTPD_ENABLED = true
2017-07-16 15:44:14 +00:00
else
2018-07-06 04:48:36 +00:00
LIGHTTPD_ENABLED = false
2017-07-16 15:44:14 +00:00
fi
2019-05-12 09:27:25 +00:00
# Create the pihole user
create_pihole_user
2019-11-14 18:52:07 +00:00
2021-08-03 22:11:22 +00:00
# Check if FTL is installed - do this early on as FTL is a hard dependency for Pi-hole
local funcOutput
funcOutput = $( get_binary_name) #Store output of get_binary_name here
local binary
binary = " pihole-FTL ${ funcOutput ##*pihole-FTL } " #binary name will be the last line of the output of get_binary_name (it always begins with pihole-FTL)
local theRest
theRest = " ${ funcOutput %pihole-FTL* } " # Print the rest of get_binary_name's output to display (cut out from first instance of "pihole-FTL")
if ! FTLdetect " ${ binary } " " ${ theRest } " ; then
printf " %b FTL Engine not installed\\n" " ${ CROSS } "
exit 1
fi
2018-07-06 04:48:36 +00:00
# Install and log everything to a file
installPihole | tee -a /proc/$$ /fd/3
2018-05-14 09:38:12 +00:00
2018-07-06 04:48:36 +00:00
# Copy the temp log file into final log location for storage
copy_to_install_log
2017-02-21 10:18:47 +00:00
2018-07-06 04:48:36 +00:00
if [ [ " ${ INSTALL_WEB_INTERFACE } " = = true ] ] ; then
# Add password to web UI if there is none
pw = ""
# If no password is set,
2021-10-05 15:40:12 +00:00
if [ [ $( grep 'WEBPASSWORD' -c " ${ setupVars } " ) = = 0 ] ] ; then
2018-07-06 04:48:36 +00:00
# generate a random password
pw = $( tr -dc _A-Z-a-z-0-9 < /dev/urandom | head -c 8)
# shellcheck disable=SC1091
2018-07-17 05:36:58 +00:00
. /opt/pihole/webpage.sh
2019-05-10 14:13:23 +00:00
echo " WEBPASSWORD= $( HashPassword " ${ pw } " ) " >> " ${ setupVars } "
2018-07-06 04:48:36 +00:00
fi
fi
2018-07-26 23:27:06 +00:00
# Check for and disable systemd-resolved-DNSStubListener before reloading resolved
2018-07-06 04:48:36 +00:00
# DNSStubListener needs to remain in place for installer to download needed files,
# so this change needs to be made after installation is complete,
2018-07-26 23:27:06 +00:00
# but before starting or resarting the dnsmasq or ftl services
disable_resolved_stublistener
2018-07-06 04:48:36 +00:00
# If the Web server was installed,
if [ [ " ${ INSTALL_WEB_SERVER } " = = true ] ] ; then
if [ [ " ${ LIGHTTPD_ENABLED } " = = true ] ] ; then
2019-01-24 08:37:03 +00:00
restart_service lighttpd
2018-07-06 04:48:36 +00:00
enable_service lighttpd
else
2018-10-31 18:14:05 +00:00
printf " %b Lighttpd is disabled, skipping service restart\\n" " ${ INFO } "
2018-07-06 04:48:36 +00:00
fi
fi
2018-10-31 18:14:05 +00:00
printf " %b Restarting services...\\n" " ${ INFO } "
2018-07-06 04:48:36 +00:00
# Start services
2018-07-26 23:27:06 +00:00
2018-08-20 20:52:22 +00:00
# Enable FTL
2018-08-16 15:52:47 +00:00
# Ensure the service is enabled before trying to start it
# Fixes a problem reported on Ubuntu 18.04 where trying to start
# the service before enabling causes installer to exit
2018-07-06 04:48:36 +00:00
enable_service pihole-FTL
2022-05-18 19:11:38 +00:00
# If this is an update from a previous Pi-hole installation
# we need to move any existing `pihole*` logs from `/var/log` to `/var/log/pihole`
2022-07-03 16:54:13 +00:00
# if /var/log/pihole.log is not a symlink (set during FTL startup) move the files
2022-05-18 19:11:38 +00:00
# can be removed with Pi-hole v6.0
# To be sure FTL is not running when we move the files we explicitly stop it here
stop_service pihole-FTL & > /dev/null
2022-07-07 20:50:41 +00:00
if [ ! -d /var/log/pihole/ ] ; then
2022-07-07 20:23:13 +00:00
mkdir /var/log/pihole/
chmod 0775 /var/log/pihole/
fi
2022-07-03 16:54:13 +00:00
# Special handling for pihole-FTL.log -> pihole/FTL.log
if [ -f /var/log/pihole-FTL.log ] && [ ! -L /var/log/pihole-FTL.log ] ; then
# /var/log/pihole-FTL.log -> /var/log/pihole/FTL.log
# /var/log/pihole-FTL.log.1 -> /var/log/pihole/FTL.log.1
# /var/log/pihole-FTL.log.2.gz -> /var/log/pihole/FTL.log.2.gz
# /var/log/pihole-FTL.log.3.gz -> /var/log/pihole/FTL.log.3.gz
# /var/log/pihole-FTL.log.4.gz -> /var/log/pihole/FTL.log.4.gz
# /var/log/pihole-FTL.log.5.gz -> /var/log/pihole/FTL.log.5.gz
for f in /var/log/pihole-FTL.log*; do mv " $f " " $( sed "s/pihole-/pihole\//" <<< " $f " ) " ; done
fi
# Remaining log files
2022-05-18 19:11:38 +00:00
if [ -f /var/log/pihole.log ] && [ ! -L /var/log/pihole.log ] ; then
2022-07-07 19:12:53 +00:00
mkdir -p /var/log/pihole/
2022-05-18 19:11:38 +00:00
mv /var/log/pihole*.* /var/log/pihole/ 2>/dev/null
fi
2019-01-24 08:37:03 +00:00
restart_service pihole-FTL
2017-09-20 07:29:11 +00:00
2018-07-06 04:48:36 +00:00
# Download and compile the aggregated block list
runGravity
2017-10-28 13:40:48 +00:00
2018-07-06 04:48:36 +00:00
# Force an update of the updatechecker
/opt/pihole/updatecheck.sh
/opt/pihole/updatecheck.sh x remote
2017-06-29 01:18:52 +00:00
2018-07-06 04:48:36 +00:00
if [ [ " ${ useUpdateVars } " = = false ] ] ; then
displayFinalMessage " ${ pw } "
2017-06-21 11:49:05 +00:00
fi
2017-01-28 15:25:02 +00:00
2017-07-07 00:25:56 +00:00
# If the Web interface was installed,
2018-04-15 01:08:16 +00:00
if [ [ " ${ INSTALL_WEB_INTERFACE } " = = true ] ] ; then
2018-07-06 04:48:36 +00:00
# If there is a password,
if ( ( ${# pw } > 0 ) ) ; then
# display the password
2018-10-31 18:14:05 +00:00
printf " %b Web Interface password: %b%s%b\\n" " ${ INFO } " " ${ COL_LIGHT_GREEN } " " ${ pw } " " ${ COL_NC } "
printf " %b This can be changed using 'pihole -a -p'\\n\\n" " ${ INFO } "
2018-07-06 04:48:36 +00:00
fi
fi
if [ [ " ${ useUpdateVars } " = = false ] ] ; then
# If the Web interface was installed,
if [ [ " ${ INSTALL_WEB_INTERFACE } " = = true ] ] ; then
2018-10-31 18:14:05 +00:00
printf " %b View the web interface at http://pi.hole/admin or http://%s/admin\\n\\n" " ${ INFO } " " ${ IPV4_ADDRESS %/* } "
2018-07-06 04:48:36 +00:00
fi
# Explain to the user how to use Pi-hole as their DNS server
2018-10-31 18:14:05 +00:00
printf " %b You may now configure your devices to use the Pi-hole as their DNS server\\n" " ${ INFO } "
[ [ -n " ${ IPV4_ADDRESS %/* } " ] ] && printf " %b Pi-hole DNS (IPv4): %s\\n" " ${ INFO } " " ${ IPV4_ADDRESS %/* } "
[ [ -n " ${ IPV6_ADDRESS } " ] ] && printf " %b Pi-hole DNS (IPv6): %s\\n" " ${ INFO } " " ${ IPV6_ADDRESS } "
2021-08-03 22:43:48 +00:00
printf " %b If you have not done so already, the above IP should be set to static.\\n" " ${ INFO } "
2018-07-06 04:48:36 +00:00
INSTALL_TYPE = "Installation"
else
INSTALL_TYPE = "Update"
fi
2017-06-29 01:18:52 +00:00
2018-07-06 04:48:36 +00:00
# Display where the log file is
2018-10-31 18:14:05 +00:00
printf "\\n %b The install log is located at: %s\\n" " ${ INFO } " " ${ installLogLoc } "
printf "%b%s Complete! %b\\n" " ${ COL_LIGHT_GREEN } " " ${ INSTALL_TYPE } " " ${ COL_NC } "
2017-07-26 13:34:40 +00:00
2018-07-06 04:48:36 +00:00
if [ [ " ${ INSTALL_TYPE } " = = "Update" ] ] ; then
2018-10-31 18:14:05 +00:00
printf "\\n"
2019-08-24 11:57:23 +00:00
" ${ PI_HOLE_BIN_DIR } " /pihole version --current
2018-07-06 04:48:36 +00:00
fi
2016-10-08 19:17:04 +00:00
}
2016-11-04 03:34:04 +00:00
if [ [ " ${ PH_TEST } " != true ] ] ; then
2018-07-06 04:48:36 +00:00
main " $@ "
2016-10-11 04:14:39 +00:00
fi