public inbox for development@lists.ipfire.org
 help / color / mirror / Atom feed
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

      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