From: jon <jon.murphy@ipfire.org>
To: development@lists.ipfire.org
Subject: Re: Proof of concept: Bridge rewrite of unbound-dhcp-leases-bridge
Date: Tue, 19 Dec 2023 16:34:15 -0600 [thread overview]
Message-ID: <03F3DF58-4729-495B-AEF3-F8E7D63BF439@ipfire.org> (raw)
In-Reply-To: <9A45BED8-7F88-417C-99CF-78C52204DA1D@ipfire.org>
[-- Attachment #1: Type: text/plain, Size: 4579 bytes --]
Michael & All,
Here is another update for the `unbound-dhcp-leases-bridge` code.
I believe I have copied all the processes done in `unbound-dhcp-leases-bridge`, so if I am missing a function please speak up.
Everything seems to be working A-OK (for me!).
- I’ve re-written much/most of the code,
- added bash functions for unbound writes & deletes to get rid of duplicate code,
- and added some error (invalid) checks.
Since the code is not parsing the dhcp-leases I believe it _may_ be fast enough for lots of dhcp clients.
But I have no way of knowing without your help!
I’ve written two "compare" scripts:
1) if you want to compare the old with new. This requires some updates to the script below.
2) or if want to compare `/etc/unbound/dhcp-leases.conf` plus `/etc/unbound/hosts.conf` to the unbound cache `unbound-control list_local_data`.
Static Hosts
————————————
> • Static hosts still need to be parsed into unbound hosts.conf file
I am looking for help with this since I dont fully understand the "read_static_hosts" section:
https://github.com/ipfire/ipfire-2.x/blob/master/config/unbound/unbound-dhcp-leases-bridge#L183-L223
- Why are the static hosts read into the "unbound-dhcp-leases-bridge" program?
- Does this section write the static hosts into the the unbound `/etc/unbound/hosts.conf` file? (I dont believe it does)
- Or are the static hosts for reference only?
Currently I do a quick grep search of the `/etc/unbound/hosts.conf` file for IP address and hostname in Line 142.
If an IP/hostname exists in the static hosts file, then I exit.
So I think this is what the unbound-dhcp-leases-bridge does…
Hostnames with Spaces
—————————————————————
> • Properly handle hostnames with spaces
I remove the spaces in the hostname
- yes, I have one device with this non-legit hostname
Is this OK?
===============================================================================
NOTE: Do not place this into a Production environment - only for Test environment.
• If placed into Test environment this current script will terminate "unbound-dhcp-leases-bridge".
• Do not run the `dhcpEvent.sh` script and "unbound-dhcp-leases-bridge" in parallel.
===============================================================================
Once again I am looking for help, feedback, thoughts and comments!
Jon
> On Dec 8, 2023, at 10:26 AM, jon <jon.murphy(a)ipfire.org> wrote:
>
> The first script did not place A & PTR records into the unbound cache and this version does!
>
> Once again I am looking for your feedback, thoughts and comments!
>
>
> Jon
>
> ===============================================================================
>
> NOTE: Do not place this into a Production environment - only for Test environment.
>
>
> Description of Proof of Concept
> ———————————————————————————————
>
> Transfers DHCP lease information to unbound DNS
>
> • Enable DHCP On Commit/Release/Expire via dhcpd.conf (or dhcpd.conf.local)
> • A new (or changed or removed) dynamic/fixed lease causes Script to run
> • Script adds A & PTR records on Commit:
> • to unbound dhcp-leases.conf file
> • to unbound cache via "unbound-control load-data"
> • Script removes A & PTR records on Release or Expiry of lease
>
>
> The Script
> ———————————
>
> • Handles dynamic and fixed leases
> • Static hosts checked for duplicates
> • Re-enabled unbound-control local_data
> • unbound reloads not needed
> • A & PTR records survive IPFire reboot, unbound restart
> • no delays from difficult dhcp-leases parsing
>
> Not completed (yet):
> • Static hosts still need to be parsed into unbound hosts.conf file
> • Properly handle hostnames with spaces
> • non-legit need to change to dashes or be removed (or completely dropped)
> • to be checked with large amount of clients
> • I’ve tested with 6 devices on test system and ~40 devices on production
>
>
> <dhcpEvent_v15.sh.txt><dhcpd.conf.local>
>
>
>
> NOTE: Do not place this into a Production environment. Only for Test environment.
>
> • If placed into Test environment this current script will terminate "unbound-dhcp-leases-bridge".
> • Do not run the `dhcpEvent.sh` script and "unbound-dhcp-leases-bridge" in parallel.
>
>
[-- Attachment #2: dhcpEvent_v25.sh.txt --]
[-- Type: text/plain, Size: 5684 bytes --]
#!/bin/bash
# DHCP commit/release/expiry status drives this process
############### following 7 lines for testing only ###############
# Kill the unbound-dhcp-leases-bridge process
pgrep -f unbound-dhcp-leases-bridge &&
kill -s SIGTERM $(pgrep -d',' -f unbound-dhcp-leases-bridge)
# chmod -v 744 /root/dhcpdconf/dhcpEvent_vN.sh
# ln -vsf /root/dhcpdconf/dhcpEvent_vN.sh /root/dhcpdconf/dhcpEvent.sh
#-------------------------------------------------------------------------
dhcpCREstatus="${1:-}"
clientIP="${2:-}"
clientHostname="${3:-}"
unboundDHCPleases="/etc/unbound/dhcp-leases.conf"
unboundStaticHosts="/etc/unbound/hosts.conf"
# time to live (seconds)
TTL="60" # time to live (seconds)
tagName="dhcpEv25" # to be "dhcpEvent"
############### Functions ###############
# write A & PTR records to unbound
#
write-unbound () {
local IPaddr
local clientName
local revIP
local recA
local recPTR
IPaddr=$1
clientName=$2
# create reverse IP for arpa name
revIP=$( echo "${IPaddr}" | /usr/bin/awk -F. '{print $4"."$3"."$2"."$1}' )
# create "A" record and "PTR" record for unbound
recA="${clientName}.${DOMAINNAME}. ${TTL} IN A ${IPaddr}"
recPTR="${revIP}.in-addr.arpa. ${TTL} IN PTR ${clientName}.${DOMAINNAME}."
# send A & PTR records to unbound and dhcp-leases file
echo -e "${recA}\n${recPTR}" | /usr/sbin/unbound-control local_datas
echo "local-data: \"${recA}\"" >> "${unboundDHCPleases}"
echo "local-data: \"${recPTR}\"" >> "${unboundDHCPleases}"
msg_log "${dhcpCREstatus}: added A & PTR records to unbound"
}
# delete A & PTR records from unbound, by IP address
#
delete_unbound () {
local IPaddr
local revIP
local fqdn
local deleteList
local IPaddr=$1
# need IP address to get FQDN and arpa name
revIP=$( echo "${IPaddr}" | /usr/bin/awk -F. '{print $4"."$3"."$2"."$1}' )
fqdn=$( /bin/grep --word-regexp --ignore-case "${IPaddr}" <<< "${ucList}" | \
/usr/bin/awk '{ print $1 }' )
# does a FQDN exist for this IP address? If yes, then remove
if [[ -n "${fqdn}" ]] ; then
# remove fqdn names from unbound AND dhcp-leases file
# look for multiple instances of fqdn (client hostname.domain)
deleteList=$( /bin/grep --word-regexp --ignore-case "${fqdn}" <<< "${ucList}" | \
/usr/bin/awk '{ print $1 }' )
# delete from unbound cache
echo "${deleteList}" | /usr/sbin/unbound-control local_datas_remove
# delete from dhcp-leases.conf file
/bin/sed --in-place "/\b${fqdn}/d" "${unboundDHCPleases}"
else
msg_log "${dhcpCREstatus}: Oops, no FQDN - exit"
return 1
fi
msg_log "${dhcpCREstatus}: references to ${fqdn} removed from unbound"
}
# send message to message log
msg_log () {
if tty --silent ; then
echo "${tagName}:" "$*"
else
/usr/bin/logger --tag "${tagName}" "$*"
fi
}
############### Main ###############
# check parameter count
if (( "$#" < 2 )) || (( $# > 3 )); then
msg_log "Illegal number of parameters - exit"
exit 1
fi
# simple check for valid IP addr ( may be too simple! )
if ! /bin/grep --quiet --extended-regexp '[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}' <<< "${clientIP}" ; then
msg_log "${clientIP} is an invalid IP address - exit"
exit
fi
# need DOMAINNAME value
eval "$(/usr/local/bin/readhash /var/ipfire/main/settings)"
# remove invalid characters in client hostname
clientHostname=$( echo "${clientHostname}" | /bin/sed 's/[^A-Za-z0-9.-]//g' )
# get existing records from unbound-control list_local_data
ucList=$( /usr/sbin/unbound-control list_local_data | /bin/grep -i "${DOMAINNAME}" | expand -t1 )
# Does this Record A already exist?
UC_RecA=$( /bin/grep --ignore-case "^${clientHostname}\.${DOMAINNAME}.*${clientIP}$" <<< "${ucList}" )
# find IP addr in unbound
UC_IP=$( /bin/grep --word-regexp --ignore-case "${clientIP}" <<< "${ucList}" )
# find client name, at start of line, in unbound (looking for record A)
UC_FQDN=$( /bin/grep --word-regexp --ignore-case "^${clientHostname}\.${DOMAINNAME}" <<< "${ucList}" )
case "${dhcpCREstatus}" in
commit)
# if client hostname is blank then exit
if [[ -z "${clientHostname}" ]] ; then
msg_log "${dhcpCREstatus}: received IP ${clientIP} but missing client hostname - exit"
exit
fi
# if client exists in static hosts then exit
if /bin/grep --word-regexp --quiet --ignore-case -e "${clientIP}" -e "${clientHostname}" "${unboundStaticHosts}"
then
msg_log "${dhcpCREstatus}: ${clientHostname} or ${clientIP} found in static hosts - exit"
exit
fi
# (ip fqdn)
if [[ -n "${UC_RecA}" ]] ; then
# existing IP and existing FQDN # ( 1 1 )
msg_log "${dhcpCREstatus}: unbound hostname & IP already exist - exit"
exit
else
if [[ -z "${UC_IP}" ]] ; then
# New IP address! - write info to unbound & dhcp-leases.conf # ( 0 0/1 )
write-unbound "${clientIP}" "${clientHostname}"
else
# Existing IP address!
if [[ -z "${UC_FQDN}" ]] ; then
# Existing IP and new FQDN
# hostname/fqdn changed - delete A & PTR records from unbound # ( 1 0 )
delete_unbound "${clientIP}"
# add new to unbound!
write-unbound "${clientIP}" "${clientHostname}"
fi
fi
fi
;;
release|expiry)
# NOTE: expiry & release are the same since I don't understand the difference :-(
if [[ -z "${clientIP}" ]] ; then
msg_log "${dhcpCREstatus}: missing client IP address - exit"
exit 1
fi
# does IP addr appear in unbound? if yes, delete lines
if [[ -n "${UC_IP}" ]] ; then
delete_unbound "${clientIP}"
else
msg_log "${dhcpCREstatus}: ${clientIP} not found in unbound - done"
fi
;;
*)
msg_log "CRE script: case = no status - exit"
exit
;;
esac
exit
prev parent reply other threads:[~2023-12-19 22:34 UTC|newest]
Thread overview: 4+ messages / expand[flat|nested] mbox.gz Atom feed top
[not found] <170138447016.81511.7036801299375036151@mail02.haj.ipfire.org>
2023-12-01 0:36 ` Jon Murphy
2023-12-02 16:03 ` Jon Murphy
2023-12-08 16:26 ` jon
2023-12-19 22:34 ` jon [this message]
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=03F3DF58-4729-495B-AEF3-F8E7D63BF439@ipfire.org \
--to=jon.murphy@ipfire.org \
--cc=development@lists.ipfire.org \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox