pi-hole/advanced/Scripts/piholeDebug.sh

444 lines
12 KiB
Bash
Raw Normal View History

#!/usr/bin/env bash
# Pi-hole: A black hole for Internet advertisements
# (c) 2015, 2016 by Jacob Salmela
# Network-wide ad blocking via your Raspberry Pi
# http://pi-hole.net
2016-09-27 03:06:31 +02:00
# Generates pihole_debug.log to be used for troubleshooting.
#
# Pi-hole is free software: you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation, either version 2 of the License, or
# (at your option) any later version.
set -o pipefail
######## GLOBAL VARS ########
DEBUG_LOG="/var/log/pihole_debug.log"
DNSMASQFILE="/etc/dnsmasq.conf"
PIHOLECONFFILE="/etc/dnsmasq.d/01-pihole.conf"
LIGHTTPDFILE="/etc/lighttpd/lighttpd.conf"
LIGHTTPDERRFILE="/var/log/lighttpd/error.log"
GRAVITYFILE="/etc/pihole/gravity.list"
HOSTSFILE="/etc/hosts"
WHITELISTFILE="/etc/pihole/whitelist.txt"
BLACKLISTFILE="/etc/pihole/blacklist.txt"
ADLISTSFILE="/etc/pihole/adlists.list"
PIHOLELOG="/var/log/pihole.log"
WHITELISTMATCHES="/tmp/whitelistmatches.list"
2016-09-27 03:06:31 +02:00
# Header info and introduction
cat << EOM
::: Beginning Pi-hole debug at $(date)!
::: This debugging process will collect information from your running configuration,
::: and optionally upload the generated log to a unique and random directory on
::: Termbin.com. NOTE: All log files auto-delete after 1 month and you are the only
::: person who is given the unique URL. Please consider where you post this link.
:::
EOM
2016-09-27 03:06:31 +02:00
# Ensure the file exists, create if not, clear if exists.
2016-10-22 09:32:36 +02:00
if [ ! -f "${DEBUG_LOG}" ]; then
touch ${DEBUG_LOG}
chmod 644 ${DEBUG_LOG}
chown "$USER":root ${DEBUG_LOG}
else
2016-08-21 03:12:02 +02:00
truncate -s 0 ${DEBUG_LOG}
fi
### Private functions exist here ###
log_write() {
2016-10-22 09:32:36 +02:00
echo "${1}" >> "${DEBUG_LOG}"
}
2016-09-27 03:06:31 +02:00
header_write() {
echo "" >> "${DEBUG_LOG}"
echo "::: ${1}" >> "${DEBUG_LOG}"
echo "" >> "${DEBUG_LOG}"
}
log_echo() {
echo "::: ${1}"
log_write "${1}"
}
version_check() {
header_write "Installed Package Versions"
echo "::: Detecting Pi-hole installed versions."
2016-09-27 03:06:31 +02:00
2016-10-22 09:32:36 +02:00
pi_hole_ver="$(cd /etc/.pihole/ && git describe --tags --abbrev=0)" \
&& log_echo "Pi-hole: $pi_hole_ver" || log_echo "Pi-hole git repository not detected."
2016-10-22 09:32:36 +02:00
admin_ver="$(cd /var/www/html/admin && git describe --tags --abbrev=0)" \
&& log_echo "WebUI: $admin_ver" || log_echo "Pi-hole Admin Pages git repository not detected."
light_ver="$(lighttpd -v |& head -n1 | cut -d " " -f1)" \
&& log_echo "${light_ver}" || log_echo "lighttpd not installed."
php_ver="$(php -v |& head -n1)" \
&& log_echo "${php_ver}" || log_echo "PHP not installed."
echo ":::"
2016-04-12 09:47:30 +02:00
}
files_check() {
header_write "Files Check"
#Check existence of setupVars.conf, and source it to get configured network interface for later use in script.
echo -n "::: Detecting existence setupVars.conf..."
setupVars=/etc/pihole/setupVars.conf
if [[ -f ${setupVars} ]];then
echo " found!"
log_write "/etc/pihole/setupVars.conf exists! Contents:"
while read -r line; do
if [ ! -z "${line}" ]; then
[[ "${line}" =~ ^#.*$ ]] && continue
log_write "${line}"
fi
done < "${setupVars}"
log_write ""
. "${setupVars}"
if [[ -n "${piholeInterface}" ]]; then
# prepend % to the beginning of piholeInterface for later use
piholeInterface="%${piholeInterface}"
fi
else
echo " NOT FOUND!"
log_write "/etc/pihole/setupVars.conf not found!"
fi
}
distro_check() {
header_write "Installed OS Distribution"
2016-09-27 03:06:31 +02:00
echo "::: Checking installed OS Distribution release."
TMP=$(cat /etc/*release || echo "Failed to find release")
2016-09-27 03:06:31 +02:00
echo "::: Writing OS Distribution release to logfile."
log_write "${TMP}"
log_write ""
2016-09-27 03:06:31 +02:00
}
ip_check() {
header_write "IP Address Information"
2016-09-27 03:06:31 +02:00
echo "::: Writing local IPs to logfile"
2016-10-22 09:32:36 +02:00
IPADDR="$(ip a | awk -F " " '{ for(i=1;i<=NF;i++) if ($i == "inet") print $(i+1) }')"
log_write "${IPADDR}"
2016-10-22 09:32:36 +02:00
IP6ADDR="$(ip a | awk -F " " '{ for(i=1;i<=NF;i++) if ($i == "inet6") print $(i+1) }')" \
&& log_write "${IP6ADDR}" || log_write "No IPv6 addresses found."
log_write ""
2016-10-22 09:32:36 +02:00
echo "::: Locating default gateway and checking connectivity"
2016-10-22 09:32:36 +02:00
GATEWAY=$(ip r | grep default | cut -d ' ' -f 3)
if [[ $? = 0 ]]; then
echo "::: Pinging default IPv4 gateway..."
2016-10-22 09:32:36 +02:00
GATEWAY_CHECK=$(ping -q -w 3 -c 3 -n "${GATEWAY}" | tail -n3)
if [[ $? = 0 ]]; then
log_write "IPv4 Gateway check:"
2016-10-22 09:32:36 +02:00
else
log_write "IPv4 Gateway check failed:"
2016-10-22 09:32:36 +02:00
fi
log_write "${GATEWAY_CHECK}"
log_write ""
2016-10-22 09:32:36 +02:00
echo "::: Pinging Internet via IPv4..."
2016-10-22 09:32:36 +02:00
INET_CHECK=$(ping -q -w 5 -c 3 -n 8.8.8.8 | tail -n3)
if [[ $? = 0 ]]; then
log_write "IPv4 Internet check:"
2016-10-22 09:32:36 +02:00
else
log_write "IPv4 Internet check failed:"
2016-10-22 09:32:36 +02:00
fi
log_write "${INET_CHECK}"
log_write ""
2016-10-22 09:32:36 +02:00
fi
GATEWAY6=$(ip -6 r | grep default | cut -d ' ' -f 3)
if [[ $? = 0 ]]; then
echo "::: Pinging default IPv6 gateway..."
GATEWAY6_CHECK=$(ping6 -q -w 3 -c 3 -n "${GATEWAY6}""${piholeInterface}" | tail -n3)
2016-10-22 09:32:36 +02:00
if [[ $? = 0 ]]; then
log_write "IPv6 Gateway check:"
2016-10-22 09:32:36 +02:00
else
log_write "IPv6 Gateway check failed:"
2016-10-22 09:32:36 +02:00
fi
echo "::: Pinging Internet via IPv6..."
GATEWAY6_CHECK=$(ping6 -q -w 3 -c 3 -n 2001:4860:4860::8888"${piholeInterface}" | tail -n3)
2016-10-22 09:32:36 +02:00
if [[ $? = 0 ]]; then
log_write "IPv6 Internet check:"
2016-10-22 09:32:36 +02:00
else
log_write "IPv6 Internet check failed:"
2016-10-22 09:32:36 +02:00
fi
else
GATEWAY_CHECK="No IPv6 Gateway Detected"
fi
log_write "${GATEWAY_CHECK}"
2016-10-22 09:32:36 +02:00
log_write ""
2016-07-15 22:11:10 +02:00
}
2016-09-27 03:52:12 +02:00
hostnameCheck() {
header_write "Hostname Information"
2016-09-27 04:23:30 +02:00
echo "::: Writing locally configured hostnames to logfile"
2016-10-22 09:32:36 +02:00
# Write the hostname output to compare against entries in /etc/hosts, which is logged next
log_write "This Pi-hole is: $(hostname)"
2016-10-22 09:32:36 +02:00
echo "::: Writing hosts file to debug log..."
log_write "### Hosts ###"
2016-10-22 09:32:36 +02:00
if [ -e "${HOSTSFILE}" ]; then
cat "${HOSTSFILE}" >> ${DEBUG_LOG}
log_write ""
2016-10-22 09:32:36 +02:00
else
log_write "No hosts file found!"
2016-10-22 09:32:36 +02:00
printf ":::\tNo hosts file found!\n"
fi
2016-09-27 03:52:12 +02:00
}
portCheck() {
header_write "Open Port Information"
2016-09-27 05:39:39 +02:00
echo "::: Detecting local server port 80 and 53 processes."
2016-09-27 05:39:39 +02:00
lsof -i :80 >> ${DEBUG_LOG}
lsof -i :53 >> ${DEBUG_LOG}
log_write ""
2016-09-27 05:39:39 +02:00
}
testResolver() {
header_write "Resolver Functions Check"
# Find a blocked url that has not been whitelisted.
2016-10-22 09:32:36 +02:00
TESTURL="doubleclick.com"
if [ -s "${WHITELISTMATCHES}" ]; then
while read -r line; do
CUTURL=${line#*" "}
2016-10-22 09:32:36 +02:00
if [ "${CUTURL}" != "Pi-Hole.IsWorking.OK" ]; then
while read -r line2; do
CUTURL2=${line2#*" "}
2016-10-22 09:32:36 +02:00
if [ "${CUTURL}" != "${CUTURL2}" ]; then
TESTURL="${CUTURL}"
break 2
fi
2016-10-22 09:32:36 +02:00
done < "${WHITELISTMATCHES}"
fi
2016-10-22 09:32:36 +02:00
done < "${GRAVITYFILE}"
fi
log_write "Resolution of ${TESTURL} from Pi-hole:"
2016-10-22 09:32:36 +02:00
LOCALDIG=$(dig "${TESTURL}" @127.0.0.1)
if [[ $? = 0 ]]; then
log_write "${LOCALDIG}"
else
log_write "Failed to resolve ${TESTURL} on Pi-hole"
fi
log_write ""
log_write "Resolution of ${TESTURL} from 8.8.8.8:"
2016-10-22 09:32:36 +02:00
REMOTEDIG=$(dig "${TESTURL}" @8.8.8.8)
if [[ $? = 0 ]]; then
log_write "${REMOTEDIG}"
else
log_write "Failed to resolve ${TESTURL} on 8.8.8.8"
2016-09-28 19:25:37 +02:00
fi
log_write ""
log_write "Pi-hole dnsmasq specific records lookups"
log_write "Cache Size:"
2016-10-22 09:32:36 +02:00
dig +short chaos txt cachesize.bind >> ${DEBUG_LOG}
log_write "Insertions count:"
2016-10-22 09:32:36 +02:00
dig +short chaos txt insertions.bind >> ${DEBUG_LOG}
log_write "Evictions count:"
2016-10-22 09:32:36 +02:00
dig +short chaos txt evictions.bind >> ${DEBUG_LOG}
log_write "Misses count:"
2016-10-22 09:32:36 +02:00
dig +short chaos txt misses.bind >> ${DEBUG_LOG}
log_write "Hits count:"
2016-10-22 09:32:36 +02:00
dig +short chaos txt hits.bind >> ${DEBUG_LOG}
log_write "Auth count:"
2016-10-22 09:32:36 +02:00
dig +short chaos txt auth.bind >> ${DEBUG_LOG}
log_write "Upstream Servers:"
2016-10-22 09:32:36 +02:00
dig +short chaos txt servers.bind >> ${DEBUG_LOG}
log_write ""
}
checkProcesses() {
header_write "Processes Check"
echo "::: Logging status of lighttpd and dnsmasq..."
PROCESSES=( lighttpd dnsmasq )
2016-10-22 09:32:36 +02:00
for i in "${PROCESSES[@]}"; do
log_write ""
log_write -n "${i}"
log_write " processes status:"
systemctl -l status "${i}" >> "${DEBUG_LOG}"
done
log_write ""
}
debugLighttpd() {
header_write "lighttpd.conf"
2016-10-22 09:32:36 +02:00
if [ -e "${LIGHTTPDFILE}" ]; then
while read -r line; do
2016-10-22 09:32:36 +02:00
if [ ! -z "${line}" ]; then
[[ "${line}" =~ ^#.*$ ]] && continue
log_write "${line}"
fi
2016-10-22 09:32:36 +02:00
done < "${LIGHTTPDFILE}"
log_write ""
else
log_write "No lighttpd.conf file found!"
printf ":::\tNo lighttpd.conf file found\n"
fi
2016-10-22 09:32:36 +02:00
if [ -e "${LIGHTTPDERRFILE}" ]; then
log_write ""
log_write "::: lighttpd error.log"
log_write ""
2016-10-22 09:32:36 +02:00
cat "${LIGHTTPDERRFILE}" >> ${DEBUG_LOG}
else
log_write "No lighttpd error.log file found!"
printf ":::\tNo lighttpd error.log file found\n"
fi
log_write ""
}
### END FUNCTIONS ###
version_check
files_check
distro_check
ip_check
2016-09-27 03:52:12 +02:00
hostnameCheck
2016-09-27 05:39:39 +02:00
portCheck
checkProcesses
testResolver
debugLighttpd
echo "::: Writing dnsmasq.conf to debug log..."
header_write "Dnsmasq configuration"
2016-10-22 09:32:36 +02:00
if [ -e "${DNSMASQFILE}" ]; then
#cat $DNSMASQFILE >> $DEBUG_LOG
while read -r line; do
2016-10-22 09:32:36 +02:00
if [ ! -z "${line}" ]; then
[[ "${line}" =~ ^#.*$ ]] && continue
log_write "${line}"
2016-10-22 09:32:36 +02:00
fi
done < "${DNSMASQFILE}"
log_write ""
else
log_write "No dnsmasq.conf file found!"
printf ":::\tNo dnsmasq.conf file found!\n"
fi
echo "::: Writing 01-pihole.conf to debug log..."
header_write "01-pihole.conf"
2016-10-22 09:32:36 +02:00
if [ -e "${PIHOLECONFFILE}" ]; then
while read -r line; do
2016-10-22 09:32:36 +02:00
if [ ! -z "${line}" ]; then
[[ "${line}" =~ ^#.*$ ]] && continue
log_write "${line}"
2016-10-22 09:32:36 +02:00
fi
done < "${PIHOLECONFFILE}"
log_write
else
log_write "No 01-pihole.conf file found!"
printf ":::\tNo 01-pihole.conf file found\n"
fi
echo "::: Writing size of gravity.list to debug log..."
header_write "gravity.list"
2016-10-22 09:32:36 +02:00
if [ -e "${GRAVITYFILE}" ]; then
wc -l "${GRAVITYFILE}" >> ${DEBUG_LOG}
log_write ""
else
log_write "No gravity.list file found!"
printf ":::\tNo gravity.list file found\n"
fi
2016-09-27 05:50:03 +02:00
### Pi-hole application specific logging ###
echo "::: Writing whitelist to debug log..."
header_write "Whitelist"
2016-10-22 09:32:36 +02:00
if [ -e "${WHITELISTFILE}" ]; then
cat "${WHITELISTFILE}" >> ${DEBUG_LOG}
log_write
else
log_write "No whitelist.txt file found!"
printf ":::\tNo whitelist.txt file found!\n"
fi
echo "::: Writing blacklist to debug log..."
header_write "Blacklist"
2016-10-22 09:32:36 +02:00
if [ -e "${BLACKLISTFILE}" ]; then
cat "${BLACKLISTFILE}" >> ${DEBUG_LOG}
log_write
else
log_write "No blacklist.txt file found!"
printf ":::\tNo blacklist.txt file found!\n"
fi
echo "::: Writing adlists.list to debug log..."
header_write "adlists.list"
2016-10-22 09:32:36 +02:00
if [ -e "${ADLISTSFILE}" ]; then
while read -r line; do
if [ ! -z "${line}" ]; then
[[ "${line}" =~ ^#.*$ ]] && continue
log_write "${line}"
2016-10-22 09:32:36 +02:00
fi
done < "${ADLISTSFILE}"
log_write
else
log_write "No adlists.list file found... using adlists.default!"
printf ":::\tNo adlists.list file found... using adlists.default!\n"
fi
# Continuously append the pihole.log file to the pihole_debug.log file
dumpPiHoleLog() {
trap '{ echo -e "\n::: Finishing debug write from interrupt... Quitting!" ; exit 1; }' INT
2016-09-27 05:50:03 +02:00
echo -e "::: Writing current Pi-hole traffic to debug log...\n:::\tTry loading any/all sites that you are having trouble with now... \n:::\t(Press ctrl+C to finish)"
header_write "pihole.log"
2016-10-22 09:32:36 +02:00
if [ -e "${PIHOLELOG}" ]; then
while true; do
2016-10-22 09:32:36 +02:00
tail -f "${PIHOLELOG}" >> ${DEBUG_LOG}
log_write ""
done
else
log_write "No pihole.log file found!"
printf ":::\tNo pihole.log file found!\n"
fi
}
# Anything to be done after capturing of pihole.log terminates
finalWork() {
2016-10-22 09:32:36 +02:00
echo "::: Finshed debugging!"
echo "::: The debug log can be uploaded to Termbin.com for easier sharing."
read -r -p "::: Would you like to upload the log? [y/N] " response
case ${response} in
[yY][eE][sS]|[yY])
TERMBIN=$(cat /var/log/pihole_debug.log | nc termbin.com 9999)
;;
*)
echo "::: Log will NOT be uploaded to Termbin."
;;
esac
# Check if termbin.com is reachable. When it's not, point to local log instead
if [ -n "${TERMBIN}" ]; then
echo "::: Debug log can be found at : ${TERMBIN}"
else
echo "::: Debug log can be found at : /var/log/pihole_debug.log"
fi
}
trap finalWork EXIT
### Method calls for additional logging ###
dumpPiHoleLog