public inbox for development@lists.ipfire.org
 help / color / mirror / Atom feed
* Re: Proof of concept: Bridge rewrite of unbound-dhcp-leases-bridge
       [not found] <170138447016.81511.7036801299375036151@mail02.haj.ipfire.org>
@ 2023-12-01  0:36 ` Jon Murphy
  2023-12-02 16:03   ` Jon Murphy
  0 siblings, 1 reply; 4+ messages in thread
From: Jon Murphy @ 2023-12-01  0:36 UTC (permalink / raw)
  To: development

[-- Attachment #1: Type: text/plain, Size: 612 bytes --]

It just dawned to me some of the things you mentioned and I may have this part wrong:
> - It appears Unbound is constantly monitoring this file.  Unbound seems to recognize the changes and accept the > A and PTR records.
>    - NOTE: I began my tests by manually entering file and they seem to be accepted fine. 
>    - NOTE: I still have testing to do to make sure this works properly!
>    - NOTE: Unbound is the part that worries me the most since I don't understand why this works!  I need to keep testing by removing/adding the unbound reload code in the existing bridge!

Back to the drawing board!

^ permalink raw reply	[flat|nested] 4+ messages in thread

* Re: Proof of concept: Bridge rewrite of unbound-dhcp-leases-bridge
  2023-12-01  0:36 ` Proof of concept: Bridge rewrite of unbound-dhcp-leases-bridge Jon Murphy
@ 2023-12-02 16:03   ` Jon Murphy
  2023-12-08 16:26     ` jon
  0 siblings, 1 reply; 4+ messages in thread
From: Jon Murphy @ 2023-12-02 16:03 UTC (permalink / raw)
  To: development

[-- Attachment #1: Type: text/plain, Size: 517 bytes --]

Michael - I believe the key to getting rid of the unbound restarts is going back to the `unbound-control local_data` type commands.

But it doesn't correct the DNS leases problem you mentioned in:  
https://bugzilla.ipfire.org/show_bug.cgi?id=13254#c27

And I believe the DNS issue and be fixed by grabbing leases differently using the On Commit/Release/Expiry status

I am going to go back to bugzilla for now.  And I'll post my updated script shortly...

https://bugzilla.ipfire.org/show_bug.cgi?id=13254

^ permalink raw reply	[flat|nested] 4+ messages in thread

* Re: Proof of concept: Bridge rewrite of unbound-dhcp-leases-bridge
  2023-12-02 16:03   ` Jon Murphy
@ 2023-12-08 16:26     ` jon
  2023-12-19 22:34       ` jon
  0 siblings, 1 reply; 4+ messages in thread
From: jon @ 2023-12-08 16:26 UTC (permalink / raw)
  To: development

[-- Attachment #1: Type: text/plain, Size: 1809 bytes --]

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









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_v15.sh.txt --]
[-- Type: text/plain, Size: 4381 bytes --]

#!/bin/bash
#set -e
#set -x

#	DHCP commit/release/expiry status drives this process

####################	following 11 lines just for testing  ####################
/usr/bin/logger --tag dhcp1 "testing --> $0 (v15)"
#	Kill the unbound-dhcp-leases-bridge process
pgrep -f unbound-dhcp-leases-bridge &&
	kill -s SIGTERM $(pgrep -d',' -f unbound-dhcp-leases-bridge) &&
	/usr/bin/logger --tag dhcp1 "terminated unbound-dhcp-leases-bridge"

#	if no dhcpLeases then create file
#[[ -f "${unboundDHCPleases}" ]] || touch "${unboundDHCPleases}"

#	chmod -v 744 /root/dhcpdconf/dhcpEvent_vN.sh
#	ln -vsf /root/dhcpdconf/dhcpEvent_vN.sh /root/dhcpdconf/dhcpEvent.sh
#-------------------------------------------------------------------------------


#	need DOMAINNAME value
eval $(/usr/local/bin/readhash /var/ipfire/main/settings)

dhcpCREstatus=$1
clientIP=$2
clientName=$3

unboundDHCPleases="/etc/unbound/dhcp-leases.conf"				#	output to unbound
unboundStaticHosts="/etc/unbound/hosts.conf"

TTL="60"														#	time to live (seconds)

#	add domain name to client name
clientFQDN=$( echo "${clientName}.${DOMAINNAME}"  )		#	 | tr '[:upper:]' '[:lower:]'

#	reverse IP
reverseIP=$( echo "${clientIP}" | awk -F. '{print $4"."$3"."$2"."$1}' )
arpaName=$( echo "${reverseIP}.in-addr.arpa" )

#	create "A" record and "PTR" record for unbound
#	made "absolute address" by adding a "." at the end of names
#	this is done for consistency with the unbound cache
aRecord="${clientFQDN}. ${TTL} IN A ${clientIP}"
ptrRecord="${arpaName}. ${TTL} IN PTR ${clientFQDN}."

#	create local data line using above Records
aRecordLD="local-data: \"${aRecord}"\"
ptrRecordLD="local-data: \"${ptrRecord}"\"

case "${dhcpCREstatus}" in
	commit)
		#	if client name is blank then exit
		[[ -z "${clientName}" ]] &&
			{ /usr/bin/logger --tag dhcp1 "${dhcpCREstatus}: no clientName - exit" ; exit ; }

		#	if client exists in static hosts then exit
		if grep --word-regexp --quiet -e "${clientIP}" -e "${clientName}" "${unboundStaticHosts}" ; then
			/usr/bin/logger --tag dhcp1 "${dhcpCREstatus}: ${clientName} or ${clientIP} found in static hosts - exit"
			exit
		fi

		#	does A record and PTR record already exist? if no, then add
		if ! grep --word-regexp --quiet -e "${clientIP}" -e "${reverseIP}" "${unboundDHCPleases}" ; then
			#	add A record and PTR record to unbound
			echo -e "${aRecordLD}\n${ptrRecordLD}" >> "${unboundDHCPleases}"
			echo -e "${aRecord}\n${ptrRecord}" | /usr/sbin/unbound-control local_datas

			/usr/bin/logger --tag dhcp1 "${dhcpCREstatus}: Record A and PTR added to unbound"
		else
			/usr/bin/logger --tag dhcp1 "${dhcpCREstatus}: Record A and PTR already exist"
		fi
		;;

	release|expiry)
		#	"expiry" and "release" are the same since I don't understand the difference    :-(

		#	does IP addr or Reverse IP appear in unbound dhcp-leases file?  if yes, delete line
		if grep --word-regexp --quiet -e "${clientIP}" -e "${reverseIP}" "${unboundDHCPleases}" ; then
			#	Since expiry does not return names, we must use the IP addr to find two FQDNs.
			#	get FQDN by searching for IP address
			uc_FQDN=$( unbound-control list_local_data | grep --word-regexp "${clientIP}" | awk '{ print $1 }' )
			[[ -z "${uc_FQDN}" ]] &&
				{ /usr/bin/logger --tag dhcp1 "${dhcpCREstatus}: Oops - no uc_FQDN - exit" ; exit ; }
				
			#	remove FQDNs from unbound dhcp-leases file (two lines in file)
			/bin/sed --in-place "/${uc_FQDN}/d" "${unboundDHCPleases}"
			/usr/bin/logger --tag dhcp1 "${dhcpCREstatus}: ${uc_FQDN} removed from ub dhcp leases"

			#	Since "local_data_remove" uses only names to delete (Client Name and arpa Name),
			#		we must get both names by searching for FQDN (returns two lines).
			#		e.g., "iMac3.localdomain."  AND  "100.6.168.192.in-addr.arpa."
			uc_Names=$( unbound-control list_local_data | awk "/${uc_FQDN}/" | awk '{ print $1 }' )

			#	for unbound cache, MUST be deleted by name for A record and arpa name for PTR record!
			echo -e "${uc_Names}" | /usr/sbin/unbound-control local_datas_remove

			/usr/bin/logger --tag dhcp1 "${dhcpCREstatus}: ${uc_Names} removed from unbound-control"
		else
			/usr/bin/logger --tag dhcp1 "${dhcpCREstatus}: ${clientIP} or ${reverseIP} not found"
		fi
		;;

	*)
		/usr/bin/logger --tag dhcp1 "CRE script: case = no status"
		echo "Usage:  dhcpEvent <status> <ip_address> <client_hostname>"
		exit
		;;
esac

exit

[-- Attachment #3: dhcpd.conf.local --]
[-- Type: application/octet-stream, Size: 1360 bytes --]

#	Do NOT delete this file!
#		it can be modified but this file cannot be deleted

#
#	commit
#
on commit {
set ClientIP = binary-to-ascii(10, 8, ".", leased-address);

#set NoName = concat("dhcp-", binary-to-ascii(10, 8, "-", leased-address));
#	if preferred, NoName can be changed to blank. and then ignored in execute script
set NoName = "";
set ClientName = pick-first-value(option host-name, config-option-host-name, client-name, NoName);

#	a MAC address can easily be added for other uses.  MAC is not needed for DNS

log(concat("CRE-Commit:  ClientIP: ", ClientIP, "  Name: ", ClientName));

execute("/root/dhcpdconf/dhcpEvent.sh", "commit", ClientIP, ClientName);
}


#
#	release
#
on release {
set ClientIP = binary-to-ascii(10, 8, ".", leased-address);

#set NoName = concat("dhcp-", binary-to-ascii(10, 8, "-", leased-address));
set NoName = "";
set ClientName = pick-first-value(option host-name, config-option-host-name, client-name, NoName);

log(concat("CRE-Release:  ClientIP: ", ClientIP, "  Name: ", ClientName));

execute("/root/dhcpdconf/dhcpEvent.sh", "release", ClientIP, ClientName);
}


#
#	expiry
#
on expiry {
set ClientIP = binary-to-ascii(10, 8, ".", leased-address);
set ClientName = "";

log(concat("CRE-Expiry:  ClientIP: ", ClientIP, "  Name: ", ClientName));

execute("/root/dhcpdconf/dhcpEvent.sh", "expiry", ClientIP, "");
}

^ permalink raw reply	[flat|nested] 4+ messages in thread

* Re: Proof of concept: Bridge rewrite of unbound-dhcp-leases-bridge
  2023-12-08 16:26     ` jon
@ 2023-12-19 22:34       ` jon
  0 siblings, 0 replies; 4+ messages in thread
From: jon @ 2023-12-19 22:34 UTC (permalink / raw)
  To: development

[-- 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

^ permalink raw reply	[flat|nested] 4+ messages in thread

end of thread, other threads:[~2023-12-19 22:34 UTC | newest]

Thread overview: 4+ messages (download: mbox.gz / follow: Atom feed)
-- links below jump to the message on this page --
     [not found] <170138447016.81511.7036801299375036151@mail02.haj.ipfire.org>
2023-12-01  0:36 ` Proof of concept: Bridge rewrite of unbound-dhcp-leases-bridge Jon Murphy
2023-12-02 16:03   ` Jon Murphy
2023-12-08 16:26     ` jon
2023-12-19 22:34       ` jon

This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox