2016-11-16 21:34:43 +01:00
|
|
|
#!/usr/bin/env bash
|
2017-09-14 16:24:29 +02:00
|
|
|
# shellcheck disable=SC1090
|
|
|
|
|
2016-11-16 21:34:43 +01:00
|
|
|
# Pi-hole: A black hole for Internet advertisements
|
2017-02-22 18:55:20 +01:00
|
|
|
# (c) 2017 Pi-hole, LLC (https://pi-hole.net)
|
|
|
|
# Network-wide ad blocking via your own hardware.
|
|
|
|
#
|
2016-11-16 22:13:47 +01:00
|
|
|
# Web interface settings
|
2016-11-16 21:34:43 +01:00
|
|
|
#
|
2017-02-22 18:55:20 +01:00
|
|
|
# This file is copyright under the latest version of the EUPL.
|
|
|
|
# Please see LICENSE file for your rights under this license.
|
|
|
|
|
2016-12-28 03:14:47 +01:00
|
|
|
readonly setupVars="/etc/pihole/setupVars.conf"
|
|
|
|
readonly dnsmasqconfig="/etc/dnsmasq.d/01-pihole.conf"
|
|
|
|
readonly dhcpconfig="/etc/dnsmasq.d/02-pihole-dhcp.conf"
|
2018-04-04 13:12:39 +02:00
|
|
|
readonly FTLconf="/etc/pihole/pihole-FTL.conf"
|
2017-01-25 10:33:25 +01:00
|
|
|
# 03 -> wildcards
|
|
|
|
readonly dhcpstaticconfig="/etc/dnsmasq.d/04-pihole-static-dhcp.conf"
|
2019-08-24 13:33:32 +02:00
|
|
|
readonly PI_HOLE_BIN_DIR="/usr/local/bin"
|
2019-10-27 02:07:08 +02:00
|
|
|
readonly dnscustomfile="/etc/pihole/custom.list"
|
2016-12-28 03:14:47 +01:00
|
|
|
|
2019-04-28 21:39:06 +02:00
|
|
|
readonly gravityDBfile="/etc/pihole/gravity.db"
|
|
|
|
|
2017-06-21 13:49:05 +02:00
|
|
|
coltable="/opt/pihole/COL_TABLE"
|
|
|
|
if [[ -f ${coltable} ]]; then
|
2018-07-20 22:57:15 +02:00
|
|
|
source ${coltable}
|
2017-06-21 13:49:05 +02:00
|
|
|
fi
|
|
|
|
|
2016-11-16 21:34:43 +01:00
|
|
|
helpFunc() {
|
2018-07-20 22:57:15 +02:00
|
|
|
echo "Usage: pihole -a [options]
|
2017-05-14 03:11:44 +02:00
|
|
|
Example: pihole -a -p password
|
|
|
|
Set options for the Admin Console
|
|
|
|
|
|
|
|
Options:
|
|
|
|
-p, password Set Admin Console password
|
|
|
|
-c, celsius Set Celsius as preferred temperature unit
|
|
|
|
-f, fahrenheit Set Fahrenheit as preferred temperature unit
|
|
|
|
-k, kelvin Set Kelvin as preferred temperature unit
|
2017-07-26 19:00:08 +02:00
|
|
|
-r, hostrecord Add a name to the DNS associated to an IPv4/IPv6 address
|
2017-09-14 16:24:29 +02:00
|
|
|
-e, email Set an administrative contact address for the Block Page
|
2017-05-14 03:11:44 +02:00
|
|
|
-h, --help Show this help dialog
|
|
|
|
-i, interface Specify dnsmasq's interface listening behavior
|
2018-12-10 16:18:25 +01:00
|
|
|
-l, privacylevel Set privacy level (0 = lowest, 4 = highest)"
|
2018-07-20 22:57:15 +02:00
|
|
|
exit 0
|
2016-11-16 21:34:43 +01:00
|
|
|
}
|
|
|
|
|
2016-12-28 03:14:47 +01:00
|
|
|
add_setting() {
|
2018-07-20 22:57:15 +02:00
|
|
|
echo "${1}=${2}" >> "${setupVars}"
|
2016-12-28 03:14:47 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
delete_setting() {
|
2018-07-20 22:57:15 +02:00
|
|
|
sed -i "/${1}/d" "${setupVars}"
|
2016-12-28 03:14:47 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
change_setting() {
|
2018-07-20 22:57:15 +02:00
|
|
|
delete_setting "${1}"
|
|
|
|
add_setting "${1}" "${2}"
|
2016-12-28 03:14:47 +01:00
|
|
|
}
|
|
|
|
|
2018-04-04 13:12:39 +02:00
|
|
|
addFTLsetting() {
|
2018-07-20 22:57:15 +02:00
|
|
|
echo "${1}=${2}" >> "${FTLconf}"
|
2018-04-04 13:12:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
deleteFTLsetting() {
|
2018-07-20 22:57:15 +02:00
|
|
|
sed -i "/${1}/d" "${FTLconf}"
|
2018-04-04 13:12:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
changeFTLsetting() {
|
2018-07-20 22:57:15 +02:00
|
|
|
deleteFTLsetting "${1}"
|
|
|
|
addFTLsetting "${1}" "${2}"
|
2018-04-04 13:12:39 +02:00
|
|
|
}
|
|
|
|
|
2016-12-28 03:14:47 +01:00
|
|
|
add_dnsmasq_setting() {
|
2018-07-20 22:57:15 +02:00
|
|
|
if [[ "${2}" != "" ]]; then
|
|
|
|
echo "${1}=${2}" >> "${dnsmasqconfig}"
|
|
|
|
else
|
|
|
|
echo "${1}" >> "${dnsmasqconfig}"
|
|
|
|
fi
|
2016-12-28 03:14:47 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
delete_dnsmasq_setting() {
|
2018-07-20 22:57:15 +02:00
|
|
|
sed -i "/${1}/d" "${dnsmasqconfig}"
|
2016-12-28 03:14:47 +01:00
|
|
|
}
|
|
|
|
|
2017-05-14 03:11:44 +02:00
|
|
|
SetTemperatureUnit() {
|
2018-07-20 22:57:15 +02:00
|
|
|
change_setting "TEMPERATUREUNIT" "${unit}"
|
|
|
|
echo -e " ${TICK} Set temperature unit to ${unit}"
|
2016-11-16 21:34:43 +01:00
|
|
|
}
|
|
|
|
|
2017-05-14 03:11:44 +02:00
|
|
|
HashPassword() {
|
2018-07-20 22:57:15 +02:00
|
|
|
# Compute password hash twice to avoid rainbow table vulnerability
|
2019-08-14 23:28:13 +02:00
|
|
|
return=$(echo -n "${1}" | sha256sum | sed 's/\s.*$//')
|
|
|
|
return=$(echo -n "${return}" | sha256sum | sed 's/\s.*$//')
|
|
|
|
echo "${return}"
|
2017-05-02 23:24:37 +02:00
|
|
|
}
|
|
|
|
|
2017-05-14 03:11:44 +02:00
|
|
|
SetWebPassword() {
|
2018-07-20 22:57:15 +02:00
|
|
|
if [ "${SUDO_USER}" == "www-data" ]; then
|
|
|
|
echo "Security measure: user www-data is not allowed to change webUI password!"
|
|
|
|
echo "Exiting"
|
|
|
|
exit 1
|
|
|
|
fi
|
|
|
|
|
|
|
|
if [ "${SUDO_USER}" == "lighttpd" ]; then
|
|
|
|
echo "Security measure: user lighttpd is not allowed to change webUI password!"
|
|
|
|
echo "Exiting"
|
|
|
|
exit 1
|
|
|
|
fi
|
|
|
|
|
|
|
|
if (( ${#args[2]} > 0 )) ; then
|
|
|
|
readonly PASSWORD="${args[2]}"
|
|
|
|
readonly CONFIRM="${PASSWORD}"
|
|
|
|
else
|
|
|
|
# Prevents a bug if the user presses Ctrl+C and it continues to hide the text typed.
|
|
|
|
# So we reset the terminal via stty if the user does press Ctrl+C
|
|
|
|
trap '{ echo -e "\nNo password will be set" ; stty sane ; exit 1; }' INT
|
2018-10-26 20:17:49 +02:00
|
|
|
read -s -r -p "Enter New Password (Blank for no password): " PASSWORD
|
2018-07-20 22:57:15 +02:00
|
|
|
echo ""
|
2017-05-04 11:43:48 +02:00
|
|
|
|
|
|
|
if [ "${PASSWORD}" == "" ]; then
|
2018-07-20 22:57:15 +02:00
|
|
|
change_setting "WEBPASSWORD" ""
|
|
|
|
echo -e " ${TICK} Password Removed"
|
|
|
|
exit 0
|
2017-05-04 11:43:48 +02:00
|
|
|
fi
|
|
|
|
|
2018-10-26 20:17:49 +02:00
|
|
|
read -s -r -p "Confirm Password: " CONFIRM
|
2017-05-04 11:43:48 +02:00
|
|
|
echo ""
|
2018-07-20 22:57:15 +02:00
|
|
|
fi
|
2017-03-27 19:37:19 +02:00
|
|
|
|
2018-07-20 22:57:15 +02:00
|
|
|
if [ "${PASSWORD}" == "${CONFIRM}" ] ; then
|
2018-10-21 02:08:22 +02:00
|
|
|
# We do not wrap this in brackets, otherwise BASH will expand any appropriate syntax
|
|
|
|
hash=$(HashPassword "$PASSWORD")
|
2018-07-20 22:57:15 +02:00
|
|
|
# Save hash to file
|
|
|
|
change_setting "WEBPASSWORD" "${hash}"
|
|
|
|
echo -e " ${TICK} New password set"
|
|
|
|
else
|
|
|
|
echo -e " ${CROSS} Passwords don't match. Your password has not been changed"
|
|
|
|
exit 1
|
|
|
|
fi
|
2016-11-16 21:34:43 +01:00
|
|
|
}
|
|
|
|
|
2016-12-28 03:14:47 +01:00
|
|
|
ProcessDNSSettings() {
|
2018-07-20 22:57:15 +02:00
|
|
|
source "${setupVars}"
|
|
|
|
|
|
|
|
delete_dnsmasq_setting "server"
|
|
|
|
|
|
|
|
COUNTER=1
|
2019-08-14 23:28:13 +02:00
|
|
|
while true ; do
|
2018-07-20 22:57:15 +02:00
|
|
|
var=PIHOLE_DNS_${COUNTER}
|
|
|
|
if [ -z "${!var}" ]; then
|
|
|
|
break;
|
|
|
|
fi
|
|
|
|
add_dnsmasq_setting "server" "${!var}"
|
2019-08-14 23:28:13 +02:00
|
|
|
(( COUNTER++ ))
|
2018-07-20 22:57:15 +02:00
|
|
|
done
|
|
|
|
|
|
|
|
# The option LOCAL_DNS_PORT is deprecated
|
|
|
|
# We apply it once more, and then convert it into the current format
|
2019-08-14 23:28:13 +02:00
|
|
|
if [ -n "${LOCAL_DNS_PORT}" ]; then
|
2018-07-20 22:57:15 +02:00
|
|
|
add_dnsmasq_setting "server" "127.0.0.1#${LOCAL_DNS_PORT}"
|
|
|
|
add_setting "PIHOLE_DNS_${COUNTER}" "127.0.0.1#${LOCAL_DNS_PORT}"
|
|
|
|
delete_setting "LOCAL_DNS_PORT"
|
|
|
|
fi
|
2018-04-18 17:12:20 +02:00
|
|
|
|
2018-07-20 22:57:15 +02:00
|
|
|
delete_dnsmasq_setting "domain-needed"
|
2016-12-28 03:14:47 +01:00
|
|
|
|
2018-07-20 22:57:15 +02:00
|
|
|
if [[ "${DNS_FQDN_REQUIRED}" == true ]]; then
|
|
|
|
add_dnsmasq_setting "domain-needed"
|
|
|
|
fi
|
2016-12-28 03:14:47 +01:00
|
|
|
|
2018-07-20 22:57:15 +02:00
|
|
|
delete_dnsmasq_setting "bogus-priv"
|
2016-12-28 03:14:47 +01:00
|
|
|
|
2018-07-20 22:57:15 +02:00
|
|
|
if [[ "${DNS_BOGUS_PRIV}" == true ]]; then
|
|
|
|
add_dnsmasq_setting "bogus-priv"
|
|
|
|
fi
|
2016-12-28 03:14:47 +01:00
|
|
|
|
2018-07-20 22:57:15 +02:00
|
|
|
delete_dnsmasq_setting "dnssec"
|
|
|
|
delete_dnsmasq_setting "trust-anchor="
|
2017-01-12 16:02:41 +01:00
|
|
|
|
2018-07-20 22:57:15 +02:00
|
|
|
if [[ "${DNSSEC}" == true ]]; then
|
|
|
|
echo "dnssec
|
2017-12-18 22:14:28 +01:00
|
|
|
trust-anchor=.,19036,8,2,49AAC11D7B6F6446702E54A1607371607A1A41855200FD2CE1CDDE32F24E8FB5
|
|
|
|
trust-anchor=.,20326,8,2,E06D44B80B8F1D39A95C0B0D7C65D08458E880409BBC683457104237C7F8EC8D
|
2017-01-12 16:02:41 +01:00
|
|
|
" >> "${dnsmasqconfig}"
|
2018-07-20 22:57:15 +02:00
|
|
|
fi
|
2017-01-12 16:02:41 +01:00
|
|
|
|
2018-07-20 22:57:15 +02:00
|
|
|
delete_dnsmasq_setting "host-record"
|
2017-02-22 14:43:07 +01:00
|
|
|
|
2019-08-14 23:28:13 +02:00
|
|
|
if [ -n "${HOSTRECORD}" ]; then
|
2018-07-20 22:57:15 +02:00
|
|
|
add_dnsmasq_setting "host-record" "${HOSTRECORD}"
|
|
|
|
fi
|
2017-02-22 14:43:07 +01:00
|
|
|
|
2018-07-20 22:57:15 +02:00
|
|
|
# Setup interface listening behavior of dnsmasq
|
|
|
|
delete_dnsmasq_setting "interface"
|
|
|
|
delete_dnsmasq_setting "local-service"
|
2017-03-01 10:46:20 +01:00
|
|
|
|
2018-07-20 22:57:15 +02:00
|
|
|
if [[ "${DNSMASQ_LISTENING}" == "all" ]]; then
|
|
|
|
# Listen on all interfaces, permit all origins
|
|
|
|
add_dnsmasq_setting "except-interface" "nonexisting"
|
|
|
|
elif [[ "${DNSMASQ_LISTENING}" == "local" ]]; then
|
|
|
|
# Listen only on all interfaces, but only local subnets
|
|
|
|
add_dnsmasq_setting "local-service"
|
|
|
|
else
|
|
|
|
# Listen only on one interface
|
|
|
|
# Use eth0 as fallback interface if interface is missing in setupVars.conf
|
|
|
|
if [ -z "${PIHOLE_INTERFACE}" ]; then
|
|
|
|
PIHOLE_INTERFACE="eth0"
|
|
|
|
fi
|
2018-01-07 14:34:02 +01:00
|
|
|
|
2018-07-20 22:57:15 +02:00
|
|
|
add_dnsmasq_setting "interface" "${PIHOLE_INTERFACE}"
|
|
|
|
fi
|
2017-03-01 10:46:20 +01:00
|
|
|
|
2018-07-20 22:57:15 +02:00
|
|
|
if [[ "${CONDITIONAL_FORWARDING}" == true ]]; then
|
|
|
|
add_dnsmasq_setting "server=/${CONDITIONAL_FORWARDING_DOMAIN}/${CONDITIONAL_FORWARDING_IP}"
|
|
|
|
add_dnsmasq_setting "server=/${CONDITIONAL_FORWARDING_REVERSE}/${CONDITIONAL_FORWARDING_IP}"
|
|
|
|
fi
|
2019-09-07 23:11:20 +02:00
|
|
|
|
|
|
|
# Prevent Firefox from automatically switching over to DNS-over-HTTPS
|
|
|
|
# This follows https://support.mozilla.org/en-US/kb/configuring-networks-disable-dns-over-https
|
|
|
|
# (sourced 7th September 2019)
|
|
|
|
add_dnsmasq_setting "server=/use-application-dns.net/"
|
2016-12-28 03:14:47 +01:00
|
|
|
}
|
|
|
|
|
2017-05-14 03:11:44 +02:00
|
|
|
SetDNSServers() {
|
2018-07-20 22:57:15 +02:00
|
|
|
# Save setting to file
|
|
|
|
delete_setting "PIHOLE_DNS"
|
|
|
|
IFS=',' read -r -a array <<< "${args[2]}"
|
|
|
|
for index in "${!array[@]}"
|
|
|
|
do
|
|
|
|
add_setting "PIHOLE_DNS_$((index+1))" "${array[index]}"
|
|
|
|
done
|
|
|
|
|
|
|
|
if [[ "${args[3]}" == "domain-needed" ]]; then
|
|
|
|
change_setting "DNS_FQDN_REQUIRED" "true"
|
|
|
|
else
|
|
|
|
change_setting "DNS_FQDN_REQUIRED" "false"
|
|
|
|
fi
|
|
|
|
|
|
|
|
if [[ "${args[4]}" == "bogus-priv" ]]; then
|
|
|
|
change_setting "DNS_BOGUS_PRIV" "true"
|
|
|
|
else
|
|
|
|
change_setting "DNS_BOGUS_PRIV" "false"
|
|
|
|
fi
|
|
|
|
|
|
|
|
if [[ "${args[5]}" == "dnssec" ]]; then
|
|
|
|
change_setting "DNSSEC" "true"
|
|
|
|
else
|
|
|
|
change_setting "DNSSEC" "false"
|
|
|
|
fi
|
|
|
|
|
|
|
|
if [[ "${args[6]}" == "conditional_forwarding" ]]; then
|
|
|
|
change_setting "CONDITIONAL_FORWARDING" "true"
|
|
|
|
change_setting "CONDITIONAL_FORWARDING_IP" "${args[7]}"
|
|
|
|
change_setting "CONDITIONAL_FORWARDING_DOMAIN" "${args[8]}"
|
|
|
|
change_setting "CONDITIONAL_FORWARDING_REVERSE" "${args[9]}"
|
|
|
|
else
|
|
|
|
change_setting "CONDITIONAL_FORWARDING" "false"
|
|
|
|
delete_setting "CONDITIONAL_FORWARDING_IP"
|
|
|
|
delete_setting "CONDITIONAL_FORWARDING_DOMAIN"
|
|
|
|
delete_setting "CONDITIONAL_FORWARDING_REVERSE"
|
|
|
|
fi
|
|
|
|
|
|
|
|
ProcessDNSSettings
|
|
|
|
|
|
|
|
# Restart dnsmasq to load new configuration
|
|
|
|
RestartDNS
|
2016-12-11 16:54:27 +01:00
|
|
|
}
|
|
|
|
|
2017-05-14 03:11:44 +02:00
|
|
|
SetExcludeDomains() {
|
2018-07-20 22:57:15 +02:00
|
|
|
change_setting "API_EXCLUDE_DOMAINS" "${args[2]}"
|
2016-12-11 19:30:04 +01:00
|
|
|
}
|
|
|
|
|
2017-05-14 03:11:44 +02:00
|
|
|
SetExcludeClients() {
|
2018-07-20 22:57:15 +02:00
|
|
|
change_setting "API_EXCLUDE_CLIENTS" "${args[2]}"
|
2016-12-11 19:30:04 +01:00
|
|
|
}
|
|
|
|
|
2017-09-08 19:33:06 +02:00
|
|
|
Poweroff(){
|
2018-07-20 22:57:15 +02:00
|
|
|
nohup bash -c "sleep 5; poweroff" &> /dev/null </dev/null &
|
2017-04-11 19:04:44 +02:00
|
|
|
}
|
|
|
|
|
2017-05-14 03:11:44 +02:00
|
|
|
Reboot() {
|
2018-07-20 22:57:15 +02:00
|
|
|
nohup bash -c "sleep 5; reboot" &> /dev/null </dev/null &
|
2016-12-11 22:33:27 +01:00
|
|
|
}
|
|
|
|
|
2017-05-14 03:11:44 +02:00
|
|
|
RestartDNS() {
|
2019-08-24 13:33:32 +02:00
|
|
|
"${PI_HOLE_BIN_DIR}"/pihole restartdns
|
2016-12-12 10:38:21 +01:00
|
|
|
}
|
|
|
|
|
2017-05-14 03:11:44 +02:00
|
|
|
SetQueryLogOptions() {
|
2018-07-20 22:57:15 +02:00
|
|
|
change_setting "API_QUERY_LOG_SHOW" "${args[2]}"
|
2016-12-28 03:14:47 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
ProcessDHCPSettings() {
|
2018-07-20 22:57:15 +02:00
|
|
|
source "${setupVars}"
|
2017-01-03 13:34:01 +01:00
|
|
|
|
2018-07-20 22:57:15 +02:00
|
|
|
if [[ "${DHCP_ACTIVE}" == "true" ]]; then
|
2018-01-07 14:36:03 +01:00
|
|
|
interface="${PIHOLE_INTERFACE}"
|
2017-01-02 10:50:59 +01:00
|
|
|
|
2017-05-14 03:11:44 +02:00
|
|
|
# Use eth0 as fallback interface
|
|
|
|
if [ -z ${interface} ]; then
|
2018-07-20 22:57:15 +02:00
|
|
|
interface="eth0"
|
2017-05-14 03:11:44 +02:00
|
|
|
fi
|
2016-12-28 03:14:47 +01:00
|
|
|
|
2017-05-14 03:11:44 +02:00
|
|
|
if [[ "${PIHOLE_DOMAIN}" == "" ]]; then
|
2018-07-20 22:57:15 +02:00
|
|
|
PIHOLE_DOMAIN="lan"
|
|
|
|
change_setting "PIHOLE_DOMAIN" "${PIHOLE_DOMAIN}"
|
2017-05-14 03:11:44 +02:00
|
|
|
fi
|
2017-01-02 11:08:54 +01:00
|
|
|
|
2017-05-14 03:11:44 +02:00
|
|
|
if [[ "${DHCP_LEASETIME}" == "0" ]]; then
|
2018-07-20 22:57:15 +02:00
|
|
|
leasetime="infinite"
|
2017-05-14 03:11:44 +02:00
|
|
|
elif [[ "${DHCP_LEASETIME}" == "" ]]; then
|
2018-07-20 22:57:15 +02:00
|
|
|
leasetime="24"
|
|
|
|
change_setting "DHCP_LEASETIME" "${leasetime}"
|
2017-07-28 20:40:13 +02:00
|
|
|
elif [[ "${DHCP_LEASETIME}" == "24h" ]]; then
|
2018-07-20 22:57:15 +02:00
|
|
|
#Installation is affected by known bug, introduced in a previous version.
|
|
|
|
#This will automatically clean up setupVars.conf and remove the unnecessary "h"
|
|
|
|
leasetime="24"
|
|
|
|
change_setting "DHCP_LEASETIME" "${leasetime}"
|
2017-05-14 03:11:44 +02:00
|
|
|
else
|
2018-07-20 22:57:15 +02:00
|
|
|
leasetime="${DHCP_LEASETIME}h"
|
2017-05-14 03:11:44 +02:00
|
|
|
fi
|
2016-12-29 15:19:44 +01:00
|
|
|
|
2017-05-14 03:11:44 +02:00
|
|
|
# Write settings to file
|
|
|
|
echo "###############################################################################
|
2016-12-28 03:14:47 +01:00
|
|
|
# DHCP SERVER CONFIG FILE AUTOMATICALLY POPULATED BY PI-HOLE WEB INTERFACE. #
|
|
|
|
# ANY CHANGES MADE TO THIS FILE WILL BE LOST ON CHANGE #
|
|
|
|
###############################################################################
|
|
|
|
dhcp-authoritative
|
2016-12-29 15:19:44 +01:00
|
|
|
dhcp-range=${DHCP_START},${DHCP_END},${leasetime}
|
2016-12-28 03:14:47 +01:00
|
|
|
dhcp-option=option:router,${DHCP_ROUTER}
|
|
|
|
dhcp-leasefile=/etc/pihole/dhcp.leases
|
2016-12-30 17:31:57 +01:00
|
|
|
#quiet-dhcp
|
2017-01-03 14:24:33 +01:00
|
|
|
" > "${dhcpconfig}"
|
2019-05-01 11:20:26 +02:00
|
|
|
chmod 644 "${dhcpconfig}"
|
2017-01-03 14:24:33 +01:00
|
|
|
|
2018-07-20 22:57:15 +02:00
|
|
|
if [[ "${PIHOLE_DOMAIN}" != "none" ]]; then
|
|
|
|
echo "domain=${PIHOLE_DOMAIN}" >> "${dhcpconfig}"
|
|
|
|
fi
|
2017-01-18 12:22:03 +01:00
|
|
|
|
2018-11-26 20:01:41 +01:00
|
|
|
# Sourced from setupVars
|
|
|
|
# shellcheck disable=SC2154
|
2018-11-25 08:39:44 +01:00
|
|
|
if [[ "${DHCP_rapid_commit}" == "true" ]]; then
|
|
|
|
echo "dhcp-rapid-commit" >> "${dhcpconfig}"
|
|
|
|
fi
|
|
|
|
|
2017-05-14 03:11:44 +02:00
|
|
|
if [[ "${DHCP_IPv6}" == "true" ]]; then
|
2018-07-20 22:57:15 +02:00
|
|
|
echo "#quiet-dhcp6
|
2016-12-28 03:14:47 +01:00
|
|
|
#enable-ra
|
|
|
|
dhcp-option=option6:dns-server,[::]
|
2016-12-29 15:26:23 +01:00
|
|
|
dhcp-range=::100,::1ff,constructor:${interface},ra-names,slaac,${leasetime}
|
2016-12-28 21:02:48 +01:00
|
|
|
ra-param=*,0,0
|
2017-01-03 14:24:33 +01:00
|
|
|
" >> "${dhcpconfig}"
|
2017-05-14 03:11:44 +02:00
|
|
|
fi
|
2016-12-28 03:14:47 +01:00
|
|
|
|
2018-07-20 22:57:15 +02:00
|
|
|
else
|
|
|
|
if [[ -f "${dhcpconfig}" ]]; then
|
|
|
|
rm "${dhcpconfig}" &> /dev/null
|
|
|
|
fi
|
|
|
|
fi
|
2016-12-12 13:15:07 +01:00
|
|
|
}
|
|
|
|
|
2017-05-14 03:11:44 +02:00
|
|
|
EnableDHCP() {
|
2018-07-20 22:57:15 +02:00
|
|
|
change_setting "DHCP_ACTIVE" "true"
|
|
|
|
change_setting "DHCP_START" "${args[2]}"
|
|
|
|
change_setting "DHCP_END" "${args[3]}"
|
|
|
|
change_setting "DHCP_ROUTER" "${args[4]}"
|
|
|
|
change_setting "DHCP_LEASETIME" "${args[5]}"
|
|
|
|
change_setting "PIHOLE_DOMAIN" "${args[6]}"
|
|
|
|
change_setting "DHCP_IPv6" "${args[7]}"
|
2018-11-25 08:39:44 +01:00
|
|
|
change_setting "DHCP_rapid_commit" "${args[8]}"
|
2016-12-12 15:34:05 +01:00
|
|
|
|
2018-07-20 22:57:15 +02:00
|
|
|
# Remove possible old setting from file
|
|
|
|
delete_dnsmasq_setting "dhcp-"
|
|
|
|
delete_dnsmasq_setting "quiet-dhcp"
|
2016-12-28 03:14:47 +01:00
|
|
|
|
2019-05-30 16:41:37 +02:00
|
|
|
# If a DHCP client claims that its name is "wpad", ignore that.
|
|
|
|
# This fixes a security hole. see CERT Vulnerability VU#598349
|
|
|
|
# We also ignore "localhost" as Windows behaves strangely if a
|
|
|
|
# device claims this host name
|
|
|
|
add_dnsmasq_setting "dhcp-name-match=set:hostname-ignore,wpad
|
|
|
|
dhcp-name-match=set:hostname-ignore,localhost
|
|
|
|
dhcp-ignore-names=tag:hostname-ignore"
|
|
|
|
|
2018-07-20 22:57:15 +02:00
|
|
|
ProcessDHCPSettings
|
2016-12-12 15:34:05 +01:00
|
|
|
|
2018-07-20 22:57:15 +02:00
|
|
|
RestartDNS
|
2016-12-12 15:34:05 +01:00
|
|
|
}
|
|
|
|
|
2017-05-14 03:11:44 +02:00
|
|
|
DisableDHCP() {
|
2018-07-20 22:57:15 +02:00
|
|
|
change_setting "DHCP_ACTIVE" "false"
|
2016-12-28 03:14:47 +01:00
|
|
|
|
2018-07-20 22:57:15 +02:00
|
|
|
# Remove possible old setting from file
|
|
|
|
delete_dnsmasq_setting "dhcp-"
|
|
|
|
delete_dnsmasq_setting "quiet-dhcp"
|
2016-12-12 15:34:05 +01:00
|
|
|
|
2018-07-20 22:57:15 +02:00
|
|
|
ProcessDHCPSettings
|
2016-12-12 15:34:05 +01:00
|
|
|
|
2018-07-20 22:57:15 +02:00
|
|
|
RestartDNS
|
2016-12-12 15:34:05 +01:00
|
|
|
}
|
|
|
|
|
2017-05-14 03:11:44 +02:00
|
|
|
SetWebUILayout() {
|
2018-07-20 22:57:15 +02:00
|
|
|
change_setting "WEBUIBOXEDLAYOUT" "${args[2]}"
|
2016-12-13 14:59:52 +01:00
|
|
|
}
|
|
|
|
|
2017-03-31 20:35:52 +02:00
|
|
|
CustomizeAdLists() {
|
2019-04-28 21:39:06 +02:00
|
|
|
local address
|
|
|
|
address="${args[3]}"
|
2018-07-20 22:57:15 +02:00
|
|
|
|
|
|
|
if [[ "${args[2]}" == "enable" ]]; then
|
2019-07-03 07:56:10 +02:00
|
|
|
sqlite3 "${gravityDBfile}" "UPDATE adlist SET enabled = 1 WHERE address = '${address}'"
|
2018-07-20 22:57:15 +02:00
|
|
|
elif [[ "${args[2]}" == "disable" ]]; then
|
2019-07-03 07:56:10 +02:00
|
|
|
sqlite3 "${gravityDBfile}" "UPDATE adlist SET enabled = 0 WHERE address = '${address}'"
|
2018-07-20 22:57:15 +02:00
|
|
|
elif [[ "${args[2]}" == "add" ]]; then
|
2019-07-03 07:56:10 +02:00
|
|
|
sqlite3 "${gravityDBfile}" "INSERT OR IGNORE INTO adlist (address) VALUES ('${address}')"
|
2018-07-20 22:57:15 +02:00
|
|
|
elif [[ "${args[2]}" == "del" ]]; then
|
2019-07-03 07:56:10 +02:00
|
|
|
sqlite3 "${gravityDBfile}" "DELETE FROM adlist WHERE address = '${address}'"
|
2018-07-20 22:57:15 +02:00
|
|
|
else
|
|
|
|
echo "Not permitted"
|
|
|
|
return 1
|
|
|
|
fi
|
2017-03-31 20:35:52 +02:00
|
|
|
}
|
|
|
|
|
2017-05-14 03:11:44 +02:00
|
|
|
SetPrivacyMode() {
|
2018-07-20 22:57:15 +02:00
|
|
|
if [[ "${args[2]}" == "true" ]]; then
|
|
|
|
change_setting "API_PRIVACY_MODE" "true"
|
|
|
|
else
|
|
|
|
change_setting "API_PRIVACY_MODE" "false"
|
|
|
|
fi
|
2016-12-27 21:27:11 +01:00
|
|
|
}
|
|
|
|
|
2016-12-15 18:55:40 +01:00
|
|
|
ResolutionSettings() {
|
2018-07-20 22:57:15 +02:00
|
|
|
typ="${args[2]}"
|
|
|
|
state="${args[3]}"
|
2016-12-15 18:55:40 +01:00
|
|
|
|
2018-07-20 22:57:15 +02:00
|
|
|
if [[ "${typ}" == "forward" ]]; then
|
|
|
|
change_setting "API_GET_UPSTREAM_DNS_HOSTNAME" "${state}"
|
|
|
|
elif [[ "${typ}" == "clients" ]]; then
|
|
|
|
change_setting "API_GET_CLIENT_HOSTNAME" "${state}"
|
|
|
|
fi
|
2016-12-16 17:33:01 +01:00
|
|
|
}
|
|
|
|
|
2017-01-25 10:33:25 +01:00
|
|
|
AddDHCPStaticAddress() {
|
2018-07-20 22:57:15 +02:00
|
|
|
mac="${args[2]}"
|
|
|
|
ip="${args[3]}"
|
|
|
|
host="${args[4]}"
|
|
|
|
|
|
|
|
if [[ "${ip}" == "noip" ]]; then
|
|
|
|
# Static host name
|
|
|
|
echo "dhcp-host=${mac},${host}" >> "${dhcpstaticconfig}"
|
|
|
|
elif [[ "${host}" == "nohost" ]]; then
|
|
|
|
# Static IP
|
|
|
|
echo "dhcp-host=${mac},${ip}" >> "${dhcpstaticconfig}"
|
|
|
|
else
|
|
|
|
# Full info given
|
|
|
|
echo "dhcp-host=${mac},${ip},${host}" >> "${dhcpstaticconfig}"
|
|
|
|
fi
|
2017-01-25 10:33:25 +01:00
|
|
|
}
|
|
|
|
|
2017-01-25 10:35:03 +01:00
|
|
|
RemoveDHCPStaticAddress() {
|
2018-07-20 22:57:15 +02:00
|
|
|
mac="${args[2]}"
|
|
|
|
sed -i "/dhcp-host=${mac}.*/d" "${dhcpstaticconfig}"
|
2017-01-25 10:35:03 +01:00
|
|
|
}
|
|
|
|
|
2017-05-14 03:11:44 +02:00
|
|
|
SetHostRecord() {
|
2018-07-20 22:57:15 +02:00
|
|
|
if [[ "${1}" == "-h" ]] || [[ "${1}" == "--help" ]]; then
|
|
|
|
echo "Usage: pihole -a hostrecord <domain> [IPv4-address],[IPv6-address]
|
2017-07-26 19:00:08 +02:00
|
|
|
Example: 'pihole -a hostrecord home.domain.com 192.168.1.1,2001:db8:a0b:12f0::1'
|
|
|
|
Add a name to the DNS associated to an IPv4/IPv6 address
|
|
|
|
|
|
|
|
Options:
|
|
|
|
\"\" Empty: Remove host record
|
|
|
|
-h, --help Show this help dialog"
|
2018-07-20 22:57:15 +02:00
|
|
|
exit 0
|
|
|
|
fi
|
2017-07-26 19:00:08 +02:00
|
|
|
|
2018-07-20 22:57:15 +02:00
|
|
|
if [[ -n "${args[3]}" ]]; then
|
|
|
|
change_setting "HOSTRECORD" "${args[2]},${args[3]}"
|
|
|
|
echo -e " ${TICK} Setting host record for ${args[2]} to ${args[3]}"
|
|
|
|
else
|
|
|
|
change_setting "HOSTRECORD" ""
|
|
|
|
echo -e " ${TICK} Removing host record"
|
|
|
|
fi
|
2017-02-22 14:43:07 +01:00
|
|
|
|
2018-07-20 22:57:15 +02:00
|
|
|
ProcessDNSSettings
|
2017-02-22 14:43:07 +01:00
|
|
|
|
2018-07-20 22:57:15 +02:00
|
|
|
# Restart dnsmasq to load new configuration
|
|
|
|
RestartDNS
|
2017-02-22 14:43:07 +01:00
|
|
|
}
|
|
|
|
|
2017-09-14 16:24:29 +02:00
|
|
|
SetAdminEmail() {
|
2018-07-20 22:57:15 +02:00
|
|
|
if [[ "${1}" == "-h" ]] || [[ "${1}" == "--help" ]]; then
|
|
|
|
echo "Usage: pihole -a email <address>
|
2017-09-14 16:24:29 +02:00
|
|
|
Example: 'pihole -a email admin@address.com'
|
|
|
|
Set an administrative contact address for the Block Page
|
|
|
|
|
|
|
|
Options:
|
|
|
|
\"\" Empty: Remove admin contact
|
|
|
|
-h, --help Show this help dialog"
|
2018-07-20 22:57:15 +02:00
|
|
|
exit 0
|
|
|
|
fi
|
2017-09-14 16:24:29 +02:00
|
|
|
|
2018-07-20 22:57:15 +02:00
|
|
|
if [[ -n "${args[2]}" ]]; then
|
|
|
|
change_setting "ADMIN_EMAIL" "${args[2]}"
|
|
|
|
echo -e " ${TICK} Setting admin contact to ${args[2]}"
|
|
|
|
else
|
|
|
|
change_setting "ADMIN_EMAIL" ""
|
|
|
|
echo -e " ${TICK} Removing admin contact"
|
|
|
|
fi
|
2017-09-14 16:24:29 +02:00
|
|
|
}
|
|
|
|
|
2017-05-14 03:11:44 +02:00
|
|
|
SetListeningMode() {
|
2018-07-20 22:57:15 +02:00
|
|
|
source "${setupVars}"
|
2017-07-07 03:44:40 +02:00
|
|
|
|
2018-07-20 22:57:15 +02:00
|
|
|
if [[ "$3" == "-h" ]] || [[ "$3" == "--help" ]]; then
|
|
|
|
echo "Usage: pihole -a -i [interface]
|
2017-05-14 03:11:44 +02:00
|
|
|
Example: 'pihole -a -i local'
|
|
|
|
Specify dnsmasq's network interface listening behavior
|
|
|
|
|
|
|
|
Interfaces:
|
|
|
|
local Listen on all interfaces, but only allow queries from
|
|
|
|
devices that are at most one hop away (local devices)
|
|
|
|
single Listen only on ${PIHOLE_INTERFACE} interface
|
|
|
|
all Listen on all interfaces, permit all origins"
|
2018-07-20 22:57:15 +02:00
|
|
|
exit 0
|
2017-05-14 03:11:44 +02:00
|
|
|
fi
|
2017-07-07 03:44:40 +02:00
|
|
|
|
2018-07-20 22:57:15 +02:00
|
|
|
if [[ "${args[2]}" == "all" ]]; then
|
2019-10-16 04:29:55 +02:00
|
|
|
echo -e " ${INFO} Listening on all interfaces, permitting all origins. Please use a firewall!"
|
2018-07-20 22:57:15 +02:00
|
|
|
change_setting "DNSMASQ_LISTENING" "all"
|
|
|
|
elif [[ "${args[2]}" == "local" ]]; then
|
2019-10-16 04:29:55 +02:00
|
|
|
echo -e " ${INFO} Listening on all interfaces, permitting origins from one hop away (LAN)"
|
2018-07-20 22:57:15 +02:00
|
|
|
change_setting "DNSMASQ_LISTENING" "local"
|
|
|
|
else
|
|
|
|
echo -e " ${INFO} Listening only on interface ${PIHOLE_INTERFACE}"
|
|
|
|
change_setting "DNSMASQ_LISTENING" "single"
|
|
|
|
fi
|
|
|
|
|
|
|
|
# Don't restart DNS server yet because other settings
|
|
|
|
# will be applied afterwards if "-web" is set
|
|
|
|
if [[ "${args[3]}" != "-web" ]]; then
|
|
|
|
ProcessDNSSettings
|
|
|
|
# Restart dnsmasq to load new configuration
|
|
|
|
RestartDNS
|
|
|
|
fi
|
2017-03-01 10:46:20 +01:00
|
|
|
}
|
|
|
|
|
2017-05-14 03:11:44 +02:00
|
|
|
Teleporter() {
|
2019-08-14 23:28:13 +02:00
|
|
|
local datetimestamp
|
2019-08-15 11:20:55 +02:00
|
|
|
datetimestamp=$(date "+%Y-%m-%d_%H-%M-%S")
|
2019-04-23 04:43:26 +02:00
|
|
|
php /var/www/html/admin/scripts/pi-hole/php/teleporter.php > "pi-hole-teleporter_${datetimestamp}.tar.gz"
|
2017-03-05 14:08:44 +01:00
|
|
|
}
|
|
|
|
|
2019-07-06 09:32:41 +02:00
|
|
|
checkDomain()
|
|
|
|
{
|
|
|
|
local domain validDomain
|
|
|
|
# Convert to lowercase
|
|
|
|
domain="${1,,}"
|
|
|
|
validDomain=$(grep -P "^((-|_)*[a-z\\d]((-|_)*[a-z\\d])*(-|_)*)(\\.(-|_)*([a-z\\d]((-|_)*[a-z\\d])*))*$" <<< "${domain}") # Valid chars check
|
|
|
|
validDomain=$(grep -P "^[^\\.]{1,63}(\\.[^\\.]{1,63})*$" <<< "${validDomain}") # Length of each label
|
|
|
|
echo "${validDomain}"
|
|
|
|
}
|
|
|
|
|
2018-08-10 18:22:55 +02:00
|
|
|
addAudit()
|
2017-06-27 10:53:24 +02:00
|
|
|
{
|
2018-08-10 18:22:55 +02:00
|
|
|
shift # skip "-a"
|
|
|
|
shift # skip "audit"
|
2019-07-06 09:32:41 +02:00
|
|
|
local domains validDomain
|
2019-07-07 21:21:56 +02:00
|
|
|
domains=""
|
2019-07-05 14:10:33 +02:00
|
|
|
for domain in "$@"
|
2018-08-10 18:22:55 +02:00
|
|
|
do
|
2019-07-09 11:41:44 +02:00
|
|
|
# Check domain to be added. Only continue if it is valid
|
2019-07-06 09:32:41 +02:00
|
|
|
validDomain="$(checkDomain "${domain}")"
|
|
|
|
if [[ -n "${validDomain}" ]]; then
|
2019-07-08 19:22:35 +02:00
|
|
|
# Put comma in between domains when there is
|
2019-07-07 21:21:56 +02:00
|
|
|
# more than one domains to be added
|
2019-07-08 19:22:35 +02:00
|
|
|
# SQL INSERT allows adding multiple rows at once using the format
|
|
|
|
## INSERT INTO table (domain) VALUES ('abc.de'),('fgh.ij'),('klm.no'),('pqr.st');
|
2019-07-07 21:21:56 +02:00
|
|
|
if [[ -n "${domains}" ]]; then
|
|
|
|
domains="${domains},"
|
|
|
|
fi
|
|
|
|
domains="${domains}('${domain}')"
|
2019-07-06 09:32:41 +02:00
|
|
|
fi
|
2018-08-10 18:22:55 +02:00
|
|
|
done
|
2019-07-09 11:41:44 +02:00
|
|
|
# Insert only the domain here. The date_added field will be
|
|
|
|
# filled with its default value (date_added = current timestamp)
|
2019-07-08 19:22:35 +02:00
|
|
|
sqlite3 "${gravityDBfile}" "INSERT INTO domain_audit (domain) VALUES ${domains};"
|
2018-08-10 18:22:55 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
clearAudit()
|
|
|
|
{
|
2019-07-08 19:22:35 +02:00
|
|
|
sqlite3 "${gravityDBfile}" "DELETE FROM domain_audit;"
|
2017-06-27 10:53:24 +02:00
|
|
|
}
|
|
|
|
|
2018-04-04 13:12:39 +02:00
|
|
|
SetPrivacyLevel() {
|
2018-08-20 23:36:16 +02:00
|
|
|
# Set privacy level. Minimum is 0, maximum is 4
|
|
|
|
if [ "${args[2]}" -ge 0 ] && [ "${args[2]}" -le 4 ]; then
|
2018-07-20 22:57:15 +02:00
|
|
|
changeFTLsetting "PRIVACYLEVEL" "${args[2]}"
|
|
|
|
fi
|
2018-04-04 13:12:39 +02:00
|
|
|
}
|
|
|
|
|
2019-10-27 02:07:08 +02:00
|
|
|
AddCustomDNSAddress() {
|
|
|
|
ip="${args[2]}"
|
|
|
|
host="${args[3]}"
|
|
|
|
echo "${ip} ${host}" >> "${dnscustomfile}"
|
|
|
|
}
|
|
|
|
|
|
|
|
RemoveCustomDNSAddress() {
|
2019-10-28 13:21:05 +01:00
|
|
|
ip="${args[2]}"
|
|
|
|
host="${args[3]}"
|
|
|
|
sed -i "/${ip} ${host}/d" "${dnscustomfile}"
|
2019-10-27 02:07:08 +02:00
|
|
|
}
|
|
|
|
|
2016-12-28 17:25:14 +01:00
|
|
|
main() {
|
2018-07-20 22:57:15 +02:00
|
|
|
args=("$@")
|
|
|
|
|
|
|
|
case "${args[1]}" in
|
|
|
|
"-p" | "password" ) SetWebPassword;;
|
|
|
|
"-c" | "celsius" ) unit="C"; SetTemperatureUnit;;
|
|
|
|
"-f" | "fahrenheit" ) unit="F"; SetTemperatureUnit;;
|
|
|
|
"-k" | "kelvin" ) unit="K"; SetTemperatureUnit;;
|
|
|
|
"setdns" ) SetDNSServers;;
|
|
|
|
"setexcludedomains" ) SetExcludeDomains;;
|
|
|
|
"setexcludeclients" ) SetExcludeClients;;
|
|
|
|
"poweroff" ) Poweroff;;
|
|
|
|
"reboot" ) Reboot;;
|
|
|
|
"restartdns" ) RestartDNS;;
|
|
|
|
"setquerylog" ) SetQueryLogOptions;;
|
|
|
|
"enabledhcp" ) EnableDHCP;;
|
|
|
|
"disabledhcp" ) DisableDHCP;;
|
|
|
|
"layout" ) SetWebUILayout;;
|
|
|
|
"-h" | "--help" ) helpFunc;;
|
|
|
|
"privacymode" ) SetPrivacyMode;;
|
|
|
|
"resolve" ) ResolutionSettings;;
|
|
|
|
"addstaticdhcp" ) AddDHCPStaticAddress;;
|
|
|
|
"removestaticdhcp" ) RemoveDHCPStaticAddress;;
|
|
|
|
"-r" | "hostrecord" ) SetHostRecord "$3";;
|
|
|
|
"-e" | "email" ) SetAdminEmail "$3";;
|
|
|
|
"-i" | "interface" ) SetListeningMode "$@";;
|
|
|
|
"-t" | "teleporter" ) Teleporter;;
|
|
|
|
"adlist" ) CustomizeAdLists;;
|
2018-08-10 18:22:55 +02:00
|
|
|
"audit" ) addAudit "$@";;
|
|
|
|
"clearaudit" ) clearAudit;;
|
2018-07-20 22:57:15 +02:00
|
|
|
"-l" | "privacylevel" ) SetPrivacyLevel;;
|
2019-10-27 02:07:08 +02:00
|
|
|
"addcustomdns" ) AddCustomDNSAddress;;
|
|
|
|
"removecustomdns" ) RemoveCustomDNSAddress;;
|
2018-07-20 22:57:15 +02:00
|
|
|
* ) helpFunc;;
|
|
|
|
esac
|
|
|
|
|
|
|
|
shift
|
|
|
|
|
|
|
|
if [[ $# = 0 ]]; then
|
|
|
|
helpFunc
|
|
|
|
fi
|
2016-12-28 17:25:14 +01:00
|
|
|
}
|