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: Fri, 08 Dec 2023 10:26:51 -0600	[thread overview]
Message-ID: <9A45BED8-7F88-417C-99CF-78C52204DA1D@ipfire.org> (raw)
In-Reply-To: <170153301528.263996.13597084183540756018@mail02.haj.ipfire.org>

[-- 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, "");
}

  reply	other threads:[~2023-12-08 16:26 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 [this message]
2023-12-19 22:34       ` jon

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=9A45BED8-7F88-417C-99CF-78C52204DA1D@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