2016-10-26 10:36:02 +02:00
|
|
|
#!/usr/bin/env bash
|
|
|
|
# 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.
|
|
|
|
#
|
2017-05-14 03:11:44 +02:00
|
|
|
# Whitelist and blacklist domains
|
2016-10-26 10:36:02 +02: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.
|
|
|
|
|
2017-05-14 03:11:44 +02:00
|
|
|
# Globals
|
2016-10-26 10:36:02 +02:00
|
|
|
basename=pihole
|
2018-02-18 21:18:52 +01:00
|
|
|
piholeDir=/etc/"${basename}"
|
2019-04-25 12:10:42 +02:00
|
|
|
gravityDBfile="${piholeDir}/gravity.db"
|
2018-08-13 18:17:14 +02:00
|
|
|
|
2016-10-26 10:36:02 +02:00
|
|
|
reload=false
|
|
|
|
addmode=true
|
|
|
|
verbose=true
|
2018-07-08 20:37:33 +02:00
|
|
|
wildcard=false
|
2016-10-26 10:36:02 +02:00
|
|
|
|
|
|
|
domList=()
|
|
|
|
|
2019-04-25 12:10:42 +02:00
|
|
|
listType=""
|
2019-04-25 12:30:38 +02:00
|
|
|
listname=""
|
|
|
|
sqlitekey=""
|
2016-10-26 10:36:02 +02:00
|
|
|
|
2019-04-25 12:30:38 +02:00
|
|
|
# shellcheck source=/opt/pihole/COL_TABLE
|
|
|
|
source "/opt/pihole/COL_TABLE"
|
2017-06-21 13:49:05 +02:00
|
|
|
|
|
|
|
|
2016-10-26 10:36:02 +02:00
|
|
|
helpFunc() {
|
2019-04-25 12:10:42 +02:00
|
|
|
if [[ "${listType}" == "whitelist" ]]; then
|
2018-07-20 21:42:11 +02:00
|
|
|
param="w"
|
2019-04-25 12:10:42 +02:00
|
|
|
type="whitelist"
|
|
|
|
elif [[ "${listType}" == "regex" && "${wildcard}" == true ]]; then
|
2018-07-20 21:42:11 +02:00
|
|
|
param="-wild"
|
2019-04-25 12:10:42 +02:00
|
|
|
type="wildcard blacklist"
|
|
|
|
elif [[ "${listType}" == "regex" ]]; then
|
2018-07-20 21:42:11 +02:00
|
|
|
param="-regex"
|
2019-04-25 12:10:42 +02:00
|
|
|
type="regex filter"
|
2018-07-20 21:42:11 +02:00
|
|
|
else
|
|
|
|
param="b"
|
2019-04-25 12:10:42 +02:00
|
|
|
type="blacklist"
|
2018-07-20 21:42:11 +02:00
|
|
|
fi
|
2017-05-14 03:11:44 +02:00
|
|
|
|
2017-05-16 02:18:32 +02:00
|
|
|
echo "Usage: pihole -${param} [options] <domain> <domain2 ...>
|
|
|
|
Example: 'pihole -${param} site.com', or 'pihole -${param} site1.com site2.com'
|
2019-04-25 12:10:42 +02:00
|
|
|
${type^} one or more domains
|
2017-05-14 03:11:44 +02:00
|
|
|
|
2017-05-16 02:23:53 +02:00
|
|
|
Options:
|
2019-04-25 12:10:42 +02:00
|
|
|
-d, --delmode Remove domain(s) from the ${type}
|
|
|
|
-nr, --noreload Update ${type} without reloading the DNS server
|
2017-05-14 03:11:44 +02:00
|
|
|
-q, --quiet Make output less verbose
|
|
|
|
-h, --help Show this help dialog
|
2017-10-07 17:29:47 +02:00
|
|
|
-l, --list Display all your ${type}listed domains
|
|
|
|
--nuke Removes all entries in a list"
|
2017-05-14 17:43:20 +02:00
|
|
|
|
2017-05-16 02:18:32 +02:00
|
|
|
exit 0
|
2016-10-26 10:36:02 +02:00
|
|
|
}
|
|
|
|
|
2016-12-02 12:57:04 +01:00
|
|
|
EscapeRegexp() {
|
2018-07-20 21:42:11 +02:00
|
|
|
# This way we may safely insert an arbitrary
|
|
|
|
# string in our regular expressions
|
|
|
|
# This sed is intentionally executed in three steps to ease maintainability
|
|
|
|
# The first sed removes any amount of leading dots
|
2019-04-25 12:30:38 +02:00
|
|
|
echo "$@" | sed 's/^\.*//' | sed "s/[]\.|$(){}?+*^]/\\\\&/g" | sed "s/\\//\\\\\//g"
|
2016-12-02 12:57:04 +01:00
|
|
|
}
|
|
|
|
|
2017-05-14 03:11:44 +02:00
|
|
|
HandleOther() {
|
2018-07-20 21:42:11 +02:00
|
|
|
# Convert to lowercase
|
|
|
|
domain="${1,,}"
|
|
|
|
|
|
|
|
# Check validity of domain (don't check for regex entries)
|
|
|
|
if [[ "${#domain}" -le 253 ]]; then
|
2019-04-25 12:10:42 +02:00
|
|
|
if [[ "${listType}" == "regex" && "${wildcard}" == false ]]; then
|
2018-07-20 21:42:11 +02:00
|
|
|
validDomain="${domain}"
|
|
|
|
else
|
|
|
|
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
|
|
|
|
fi
|
|
|
|
fi
|
2017-05-14 03:11:44 +02:00
|
|
|
|
2018-07-20 21:42:11 +02:00
|
|
|
if [[ -n "${validDomain}" ]]; then
|
|
|
|
domList=("${domList[@]}" ${validDomain})
|
2018-06-29 05:21:01 +02:00
|
|
|
else
|
2018-07-20 21:42:11 +02:00
|
|
|
echo -e " ${CROSS} ${domain} is not a valid argument or domain name!"
|
2018-06-29 05:21:01 +02:00
|
|
|
fi
|
2016-10-26 10:36:02 +02:00
|
|
|
}
|
|
|
|
|
2019-04-25 12:10:42 +02:00
|
|
|
ProcessDomainList() {
|
2019-04-25 12:30:38 +02:00
|
|
|
if [[ "${listType}" == "regex" ]]; then
|
|
|
|
# Regex filter list
|
|
|
|
listname="regex filters"
|
|
|
|
sqlitekey="filter"
|
|
|
|
else
|
|
|
|
# Whitelist / Blacklist
|
|
|
|
listname="${listType}"
|
|
|
|
sqlitekey="domain"
|
|
|
|
fi
|
|
|
|
|
2018-07-20 21:42:11 +02:00
|
|
|
for dom in "${domList[@]}"; do
|
2019-04-25 12:30:38 +02:00
|
|
|
# Format domain into regex filter if requested
|
|
|
|
if [[ "${wildcard}" == true ]]; then
|
|
|
|
dom="(^|\\.)${dom//\./\\.}$"
|
|
|
|
fi
|
|
|
|
|
|
|
|
# Logic: If addmode then add to desired list and remove from the other; if delmode then remove from desired list but do not add to the othe
|
2018-07-20 21:42:11 +02:00
|
|
|
if ${addmode}; then
|
2019-04-25 12:10:42 +02:00
|
|
|
AddDomain "${dom}" "${listType}"
|
2019-04-25 12:30:38 +02:00
|
|
|
if [[ ! "${listType}" == "regex" ]]; then
|
|
|
|
RemoveDomain "${dom}" "${listAlt}"
|
|
|
|
fi
|
2018-07-20 21:42:11 +02:00
|
|
|
else
|
2019-04-25 12:10:42 +02:00
|
|
|
RemoveDomain "${dom}" "${listType}"
|
2018-07-20 21:42:11 +02:00
|
|
|
fi
|
2017-05-14 03:11:44 +02:00
|
|
|
done
|
2016-10-26 10:36:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
AddDomain() {
|
2019-04-25 12:30:38 +02:00
|
|
|
local domain list num
|
2019-04-25 12:10:42 +02:00
|
|
|
domain="$1"
|
2018-07-20 21:42:11 +02:00
|
|
|
list="$2"
|
|
|
|
|
2019-04-25 12:10:42 +02:00
|
|
|
# Is the domain in the list we want to add it to?
|
|
|
|
num="$(sqlite3 "${gravityDBfile}" "SELECT COUNT(*) FROM ${list} WHERE ${sqlitekey} = \"${domain}\";")"
|
2018-07-20 21:42:11 +02:00
|
|
|
|
2019-04-25 12:10:42 +02:00
|
|
|
if [[ "${num}" -eq 0 ]]; then
|
|
|
|
# Domain not found in the file, add it!
|
|
|
|
if [[ "${verbose}" == true ]]; then
|
|
|
|
echo -e " ${INFO} Adding ${1} to ${listname}..."
|
|
|
|
fi
|
|
|
|
reload=true
|
|
|
|
# Add it to the list we want to add it to
|
|
|
|
local timestamp
|
|
|
|
timestamp="$(date --utc +'%s')"
|
|
|
|
sqlite3 "${gravityDBfile}" "INSERT INTO ${list} (${sqlitekey},enabled,date_added) VALUES (\"${domain}\",1,${timestamp});"
|
|
|
|
else
|
|
|
|
if [[ "${verbose}" == true ]]; then
|
|
|
|
echo -e " ${INFO} ${1} already exists in ${listname}, no need to add!"
|
2018-07-20 21:42:11 +02:00
|
|
|
fi
|
2017-05-14 03:11:44 +02:00
|
|
|
fi
|
2016-10-26 10:36:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
RemoveDomain() {
|
2019-04-25 12:30:38 +02:00
|
|
|
local domain list num
|
2019-04-25 12:10:42 +02:00
|
|
|
domain="$1"
|
2018-07-20 21:42:11 +02:00
|
|
|
list="$2"
|
2019-04-25 12:10:42 +02:00
|
|
|
|
|
|
|
# Is the domain in the list we want to remove it from?
|
|
|
|
num="$(sqlite3 "${gravityDBfile}" "SELECT COUNT(*) FROM ${list} WHERE ${sqlitekey} = \"${domain}\";")"
|
|
|
|
|
|
|
|
if [[ "${num}" -ne 0 ]]; then
|
|
|
|
# Domain found in the file, remove it!
|
|
|
|
if [[ "${verbose}" == true ]]; then
|
|
|
|
echo -e " ${INFO} Removing ${1} from ${listname}..."
|
|
|
|
fi
|
|
|
|
reload=true
|
|
|
|
# Remove it from the current list
|
|
|
|
local timestamp
|
|
|
|
timestamp="$(date --utc +'%s')"
|
|
|
|
sqlite3 "${gravityDBfile}" "DELETE FROM ${list} WHERE ${sqlitekey} = \"${domain}\";"
|
|
|
|
else
|
|
|
|
if [[ "${verbose}" == true ]]; then
|
|
|
|
echo -e " ${INFO} ${1} does not exist in ${listname}, no need to remove!"
|
|
|
|
fi
|
|
|
|
fi
|
2016-10-26 10:36:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
Displaylist() {
|
2019-04-25 12:30:38 +02:00
|
|
|
local list listname count num_pipes domain enabled status
|
2019-04-25 12:10:42 +02:00
|
|
|
|
|
|
|
if [[ "${listType}" == "regex" ]]; then
|
|
|
|
listname="regex filters list"
|
|
|
|
else
|
|
|
|
# Whitelist / Blacklist
|
|
|
|
listname="${listType}"
|
|
|
|
fi
|
|
|
|
data="$(sqlite3 "${gravityDBfile}" "SELECT * FROM ${listType};" 2> /dev/null)"
|
|
|
|
|
|
|
|
if [[ -z $data ]]; then
|
|
|
|
echo -e "Not showing empty ${listname}"
|
|
|
|
else
|
|
|
|
echo -e "Displaying ${listname}:"
|
2018-07-20 21:42:11 +02:00
|
|
|
count=1
|
2019-04-25 12:10:42 +02:00
|
|
|
while IFS= read -r line
|
|
|
|
do
|
2019-04-25 12:30:38 +02:00
|
|
|
# Count number of pipes seen in this line
|
|
|
|
# This is necessary because we can only detect the pipe separating the fields
|
|
|
|
# from the end backwards as the domain (which is the first field) may contain
|
|
|
|
# pipe symbols as they are perfectly valid regex filter control characters
|
|
|
|
num_pipes="$(grep -c "^" <<< "$(grep -o "|" <<< "${line}")")"
|
|
|
|
|
|
|
|
domain="$(cut -d'|' -f"-$((num_pipes-2))" <<< "${line}")"
|
|
|
|
enabled="$(cut -d'|' -f$((num_pipes-1)) <<< "${line}")"
|
2019-04-25 12:10:42 +02:00
|
|
|
if [[ "${enabled}" -eq 1 ]]; then
|
|
|
|
status="enabled"
|
|
|
|
else
|
|
|
|
status="disabled"
|
|
|
|
fi
|
2019-04-25 12:30:38 +02:00
|
|
|
|
2019-04-25 12:10:42 +02:00
|
|
|
echo " ${count}: ${domain} (${status})"
|
2018-07-20 21:42:11 +02:00
|
|
|
count=$((count+1))
|
2019-04-25 12:10:42 +02:00
|
|
|
done <<< "${data}"
|
|
|
|
exit 0;
|
2017-06-21 13:49:05 +02:00
|
|
|
fi
|
2016-10-26 10:36:02 +02:00
|
|
|
}
|
|
|
|
|
2017-10-07 17:29:47 +02:00
|
|
|
NukeList() {
|
2019-04-25 12:10:42 +02:00
|
|
|
sqlite3 "${gravityDBfile}" "DELETE FROM ${listType};"
|
2017-10-07 17:29:47 +02:00
|
|
|
}
|
|
|
|
|
2016-10-26 10:36:02 +02:00
|
|
|
for var in "$@"; do
|
2018-07-20 21:42:11 +02:00
|
|
|
case "${var}" in
|
2019-04-25 12:10:42 +02:00
|
|
|
"-w" | "whitelist" ) listType="whitelist"; listAlt="blacklist";;
|
|
|
|
"-b" | "blacklist" ) listType="blacklist"; listAlt="whitelist";;
|
|
|
|
"--wild" | "wildcard" ) listType="regex"; wildcard=true;;
|
|
|
|
"--regex" | "regex" ) listType="regex";;
|
2018-07-20 21:42:11 +02:00
|
|
|
"-nr"| "--noreload" ) reload=false;;
|
|
|
|
"-d" | "--delmode" ) addmode=false;;
|
|
|
|
"-q" | "--quiet" ) verbose=false;;
|
|
|
|
"-h" | "--help" ) helpFunc;;
|
|
|
|
"-l" | "--list" ) Displaylist;;
|
|
|
|
"--nuke" ) NukeList;;
|
|
|
|
* ) HandleOther "${var}";;
|
|
|
|
esac
|
2016-10-26 10:36:02 +02:00
|
|
|
done
|
|
|
|
|
|
|
|
shift
|
|
|
|
|
|
|
|
if [[ $# = 0 ]]; then
|
2018-07-20 21:42:11 +02:00
|
|
|
helpFunc
|
2016-10-26 10:36:02 +02:00
|
|
|
fi
|
|
|
|
|
2019-04-25 12:10:42 +02:00
|
|
|
ProcessDomainList
|
2016-10-26 10:36:02 +02:00
|
|
|
|
2017-07-24 13:26:39 +02:00
|
|
|
if [[ "${reload}" != false ]]; then
|
2019-04-25 12:10:42 +02:00
|
|
|
pihole restartdns reload
|
2016-10-26 10:36:02 +02:00
|
|
|
fi
|