pi-hole/advanced/Scripts/piholeDebug.sh

460 lines
13 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 ########
2016-10-28 23:52:08 +02:00
VARSFILE="/etc/pihole/setupVars.conf"
DEBUG_LOG="/var/log/pihole_debug.log"
DNSMASQFILE="/etc/dnsmasq.conf"
2016-10-28 23:52:08 +02:00
DNSMASQCONFFILE="/etc/dnsmasq.d/01-pihole.conf"
LIGHTTPDFILE="/etc/lighttpd/lighttpd.conf"
LIGHTTPDERRFILE="/var/log/lighttpd/error.log"
GRAVITYFILE="/etc/pihole/gravity.list"
WHITELISTFILE="/etc/pihole/whitelist.txt"
BLACKLISTFILE="/etc/pihole/blacklist.txt"
2016-10-28 23:52:08 +02:00
ADLISTFILE="/etc/pihole/adlists.list"
PIHOLELOG="/var/log/pihole.log"
WHITELISTMATCHES="/tmp/whitelistmatches.list"
2016-10-28 23:52:08 +02:00
IPV6_READY=false
2016-10-26 21:22:20 +02:00
2016-09-27 03:06:31 +02:00
# Header info and introduction
cat << EOM
::: Beginning Pi-hole debug at $(date)!
2016-10-28 13:42:45 +02:00
:::
2016-10-28 23:52:08 +02:00
::: This process collects information from your Pi-hole, and optionally uploads
::: it to a unique and random directory on tricorder.pi-hole.net.
2016-10-28 13:42:45 +02:00
:::
2016-10-28 23:52:08 +02:00
::: NOTE: All log files auto-delete after 24 hours and ONLY the Pi-hole developers
::: can access your data via the given token. We have taken these extra steps to
::: secure your data and will work to further reduce any personal information gathered.
:::
2016-10-28 23:52:08 +02:00
::: Please read and note any issues, and follow any directions advised during this process.
EOM
2016-09-27 03:06:31 +02:00
# Ensure the file exists, create if not, clear if exists.
2016-10-28 23:52:08 +02:00
truncate --size=0 "${DEBUG_LOG}"
chmod 644 ${DEBUG_LOG}
chown "$USER":pihole ${DEBUG_LOG}
source ${VARSFILE}
### Private functions exist here ###
log_write() {
echo "${1}" >> "${DEBUG_LOG}"
}
2016-09-27 03:06:31 +02:00
log_echo() {
case ${1} in
-n)
echo -n "::: ${2}"
log_write "${2}"
;;
2016-10-28 23:52:08 +02:00
-r)
echo "::: ${2}"
log_write "${2}"
;;
-l)
echo "${2}"
log_write "${2}"
;;
2016-10-28 23:52:08 +02:00
-e)
echo "${2}"
log_write
;;
*)
2016-10-28 23:52:08 +02:00
echo "::: ${1}"
log_write "${1}"
esac
}
2016-10-28 23:52:08 +02:00
header_write() {
log_echo ""
log_echo "${1}"
log_write ""
}
file_parse() {
while read -r line; do
if [ ! -z "${line}" ]; then
[[ "${line}" =~ ^#.*$ ]] && continue
log_write "${line}"
fi
done < "${1}"
2016-10-28 23:52:08 +02:00
log_write ""
}
block_parse() {
log_write "${1}"
}
lsof_parse() {
local user
local process
local match
user=$(echo ${1} | cut -f 3 -d ' ' | cut -c 2-)
process=$(echo ${1} | cut -f 2 -d ' ' | cut -c 2-)
if [[ ${2} -eq ${process} ]]; then
2016-10-28 23:52:08 +02:00
echo "::: Correctly configured."
else
2016-10-28 23:52:08 +02:00
log_echo "::: Failure: Incorrectly configured daemon."
fi
2016-10-28 23:52:08 +02:00
log_write "Found user ${user} with process ${process}"
}
2016-10-28 15:51:30 +02:00
version_check() {
2016-10-28 23:52:08 +02:00
header_write "Detecting Installed Package Versions:"
local error_found
error_found=0
local pi_hole_ver="$(cd /etc/.pihole/ && git describe --tags --abbrev=0)" \
2016-10-28 23:52:08 +02:00
&& log_echo -r "Pi-hole: $pi_hole_ver" || (log_echo "Pi-hole git repository not detected." && error_found=1)
local admin_ver="$(cd /var/www/html/admin && git describe --tags --abbrev=0)" \
2016-10-28 23:52:08 +02:00
&& log_echo -r "WebUI: $admin_ver" || (log_echo "Pi-hole Admin Pages git repository not detected." && error_found=1)
local light_ver="$(lighttpd -v |& head -n1 | cut -d " " -f1)" \
2016-10-28 23:52:08 +02:00
&& log_echo -r "${light_ver}" || (log_echo "lighttpd not installed." && error_found=1)
local php_ver="$(php -v |& head -n1)" \
2016-10-28 23:52:08 +02:00
&& log_echo -r "${php_ver}" || (log_echo "PHP not installed." && error_found=1)
return "${error_found}"
2016-04-12 09:47:30 +02:00
}
files_check() {
#Check non-zero length existence of ${1}
2016-10-28 23:52:08 +02:00
header_write "Detecting existence of ${1}:"
local search_file="${1}"
if [[ -s ${search_file} ]]; then
2016-10-28 23:52:08 +02:00
echo "::: File exists"
2016-10-28 15:51:30 +02:00
file_parse "${search_file}"
return 0
else
log_echo "${1} not found!"
2016-10-28 15:51:30 +02:00
return 1
fi
echo ":::"
}
2016-10-28 15:51:30 +02:00
source_file() {
local file_found=$(files_check "${1}") \
2016-10-28 23:52:08 +02:00
&& (source "${1}" &> /dev/null && echo "${file_found} and was successfully sourced") \
2016-10-28 15:51:30 +02:00
|| log_echo -l "${file_found} and could not be sourced"
}
distro_check() {
2016-10-28 23:52:08 +02:00
local soft_fail
header_write "Detecting installed OS Distribution"
soft_fail=0
local distro="$(cat /etc/*release)" && block_parse "${distro}" || (log_echo "Distribution details not found." && soft_fail=1)
return "${soft_fail}"
}
2016-09-27 03:06:31 +02:00
2016-10-28 23:52:08 +02:00
processor_check() {
local soft_fail
header_write "Checking processor variety"
soft_fail=0
log_write $(uname -m) || soft_fail=1
return "${soft_fail}"
2016-09-27 03:06:31 +02:00
}
2016-10-26 21:22:20 +02:00
ipv6_check() {
# Check if system is IPv6 enabled, for use in other functions
2016-10-28 23:52:08 +02:00
if [[ $IPv6_address ]]; then
ls /proc/net/if_inet6 &>/dev/null && IPV6_READY=true
2016-10-26 21:22:20 +02:00
return 0
else
return 1
fi
}
ip_check() {
header_write "IP Address Information"
# Get the current interface for Internet traffic
2016-10-26 21:22:20 +02:00
# Check if IPv6 enabled
local IPv6_interface
ipv6_check && IPv6_interface=${piholeInterface:-$(ip -6 r | grep default | cut -d ' ' -f 5)}
# If declared in setupVars.conf use it, otherwise defer to default
2016-10-26 06:53:41 +02:00
# http://stackoverflow.com/questions/2013547/assigning-default-values-to-shell-variables-with-a-single-command-in-bash
2016-09-27 03:06:31 +02:00
2016-10-28 23:52:08 +02:00
local IPv4_addr_list="$(ip a | awk -F " " '{ for(i=1;i<=NF;i++) if ($i == "inet") print $(i+1) }')" \
&& (block_parse "${IPv4_addr_list}" && echo "::: IPv4 addresses located")\
|| log_echo "No IPv4 addresses found."
local IPv4_def_gateway=$(ip r | grep default | cut -d ' ' -f 3)
2016-10-22 09:32:36 +02:00
if [[ $? = 0 ]]; then
2016-10-28 23:52:08 +02:00
echo -n "::: Pinging default IPv4 gateway: "
local IPv4_def_gateway_check="$(ping -q -w 3 -c 3 -n "${IPv4_def_gateway}" | tail -n3)" \
&& echo "Gateway responded." \
|| echo "Gateway did not respond."
block_parse "${IPv4_def_gateway_check}"
2016-10-22 09:32:36 +02:00
2016-10-28 23:52:08 +02:00
echo -n "::: Pinging Internet via IPv4: "
local IPv4_inet_check="$(ping -q -w 5 -c 3 -n 8.8.8.8 | tail -n3)" \
&& echo "Query responded." \
|| echo "Query did not respond."
block_parse "${IPv4_inet_check}"
2016-10-22 09:32:36 +02:00
fi
2016-10-28 23:52:08 +02:00
if [[ IPV6_READY ]]; then
2016-10-26 21:22:20 +02:00
local IPv6_addr_list="$(ip a | awk -F " " '{ for(i=1;i<=NF;i++) if ($i == "inet6") print $(i+1) }')" \
&& (log_write "${IPv6_addr_list}" && echo "::: IPv6 addresses located") \
|| log_echo "No IPv6 addresses found."
local IPv6_def_gateway=$(ip -6 r | grep default | cut -d ' ' -f 3)
2016-10-26 22:57:57 +02:00
if [[ $? = 0 ]] && [[ -n ${IPv6_def_gateway} ]]; then
2016-10-28 23:52:08 +02:00
echo -n "::: Pinging default IPv6 gateway: "
2016-10-26 21:22:20 +02:00
local IPv6_def_gateway_check="$(ping6 -q -W 3 -c 3 -n "${IPv6_def_gateway}" -I "${IPv6_interface}"| tail -n3)" \
&& echo "Gateway Responded." \
|| echo "Gateway did not respond."
block_parse "${IPv6_def_gateway_check}"
2016-10-26 21:22:20 +02:00
2016-10-28 23:52:08 +02:00
echo -n "::: Pinging Internet via IPv6: "
2016-10-26 21:22:20 +02:00
local IPv6_inet_check=$(ping6 -q -W 3 -c 3 -n 2001:4860:4860::8888 -I "${IPv6_interface}"| tail -n3) \
&& echo "Query responded." \
|| echo "Query did not respond."
block_parse "${IPv6_inet_check}"
2016-10-26 21:22:20 +02:00
else
log_echo="No IPv6 Gateway Detected"
2016-10-26 21:22:20 +02:00
fi
fi
2016-07-15 22:11:10 +02:00
}
2016-09-27 03:52:12 +02:00
2016-10-28 23:52:08 +02:00
port_check() {
local lsof_value
2016-10-22 09:32:36 +02:00
2016-10-28 23:52:08 +02:00
lsof_value=$(lsof -i ${1}:${2} -FcL | tr '\n' ' ') \
&& lsof_parse "${lsof_value}" "${3}" \
|| log_echo "Failure: IPv${1} Port not in use"
}
2016-10-26 20:38:19 +02:00
daemon_check() {
# Check for daemon ${1} on port ${2}
header_write "Daemon Process Information"
2016-10-26 20:38:19 +02:00
2016-10-28 23:52:08 +02:00
echo "::: Checking ${2} port for ${1} listener."
2016-10-28 23:52:08 +02:00
if [[ ${IPV6_READY} ]]; then
port_check 6 "${2}" "${1}"
fi
2016-10-27 09:30:51 +02:00
lsof_value=$(lsof -i 4:${2} -FcL | tr '\n' ' ') \
2016-10-28 23:52:08 +02:00
port_check 4 "${2}" "${1}"
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 "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 "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() {
2016-10-28 15:51:30 +02:00
echo "::: Checking for necessary lighttpd files."
files_check "${LIGHTTPDFILE}"
files_check "${LIGHTTPDERRFILE}"
echo ":::"
}
### END FUNCTIONS ###
# Gather version of required packages / repositories
version_check || echo "REQUIRED FILES MISSING"
2016-10-28 23:52:08 +02:00
# Check for newer setupVars storage file
2016-10-28 15:51:30 +02:00
source_file "/etc/pihole/setupVars.conf"
2016-10-28 23:52:08 +02:00
# Gather information about the running distribution
distro_check || echo "Distro Check soft fail"
# Gather processor type
processor_check || echo "Processor Check soft fail"
ip_check
#hostnameCheck
2016-10-26 20:38:19 +02:00
daemon_check lighttpd http
daemon_check dnsmasq domain
checkProcesses
testResolver
debugLighttpd
header_write "Dnsmasq configuration"
2016-10-28 23:52:08 +02:00
files_check ${DNSMASQFILE}
echo "::: Writing 01-pihole.conf to debug log..."
header_write "01-pihole.conf"
2016-10-28 23:52:08 +02:00
if [ -e "${DNSMASQCONFFILE}" ]; 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
2016-10-28 23:52:08 +02:00
done < "${DNSMASQCONFFILE}"
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-28 23:52:08 +02:00
if [ -e "${ADLISTFILE}" ]; then
2016-10-22 09:32:36 +02:00
while read -r line; do
if [ ! -z "${line}" ]; then
[[ "${line}" =~ ^#.*$ ]] && continue
log_write "${line}"
2016-10-22 09:32:36 +02:00
fi
2016-10-28 23:52:08 +02:00
done < "${ADLISTFILE}"
log_write
else
log_write "No adlists.list file found... using adlists.default!"
printf ":::\tNo adlists.list file found... using adlists.default!\n"
fi
echo
# 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-28 13:42:45 +02:00
local tricorder
2016-10-22 09:32:36 +02:00
echo "::: Finshed debugging!"
2016-10-28 13:42:45 +02:00
echo "::: The debug log can be uploaded to tricorder.pi-hole.net for sharing with developers only."
2016-10-22 09:32:36 +02:00
read -r -p "::: Would you like to upload the log? [y/N] " response
case ${response} in
[yY][eE][sS]|[yY])
2016-10-28 13:42:45 +02:00
tricorder=$(cat /var/log/pihole_debug.log | nc tricorder.pi-hole.net 9999)
2016-10-22 09:32:36 +02:00
;;
*)
2016-10-28 13:42:45 +02:00
echo "::: Log will NOT be uploaded to tricorder."
2016-10-22 09:32:36 +02:00
;;
esac
2016-10-28 13:42:45 +02:00
# Check if tricorder.pi-hole.net is reachable and provide token.
if [ -n "${tricorder}" ]; then
echo "::: Your debug token is : ${tricorder}"
2016-10-22 09:32:36 +02:00
fi
2016-10-28 13:42:45 +02:00
echo "::: Debug log can be found at : /var/log/pihole_debug.log"
}
trap finalWork EXIT
### Method calls for additional logging ###
dumpPiHoleLog