Files
notes-techniques/scripts/server-dhcp/get_first_free_ip.sh
2025-03-14 08:12:48 +01:00

82 lines
2.5 KiB
Bash

#!/bin/bash
# Basé sur un travail de Cédric Abonnel / Cédrix sous licence CC BY-NC 4.0
# Déterminer l'interface réseau active (exclure loopback)
INTERFACE=$(ip route | awk '/default/ {print $5; exit}')
# Vérifier si une interface est détectée
if [[ -z "$INTERFACE" ]]; then
echo "❌ Aucune interface réseau active détectée."
exit 1
fi
# Déterminer l'adresse IP et le masque réseau de l'interface active
IP_INFO=$(ip -4 addr show "$INTERFACE" | awk '/inet / {print $2}')
NETWORK_IP=$(echo "$IP_INFO" | cut -d'/' -f1)
NETMASK_CIDR=$(echo "$IP_INFO" | cut -d'/' -f2)
# Extraire le préfixe réseau (ex: "192.168.100")
PREFIX=$(echo "$NETWORK_IP" | awk -F. '{print $1"."$2"."$3}')
# Calculer les bornes de la plage d'adresses IP en fonction du masque
NETMASK=$(( 32 - NETMASK_CIDR ))
TOTAL_HOSTS=$(( 2 ** NETMASK ))
IP_RANGE_START="$PREFIX.1"
IP_RANGE_END="$PREFIX.$(( TOTAL_HOSTS - 2 ))" # -2 pour exclure le broadcast
# Fichier contenant les réservations DHCP
RESERVATION_FILE="/etc/dnsmasq.d/custom_hosts"
# Fonction pour convertir une IP en un entier
ip_to_int() {
local IFS=.
local ip=($1)
echo $(( (ip[0] << 24) + (ip[1] << 16) + (ip[2] << 8) + ip[3] ))
}
# Fonction pour convertir un entier en IP
int_to_ip() {
local ip=$1
echo "$(( (ip >> 24) & 255 )).$(( (ip >> 16) & 255 )).$(( (ip >> 8) & 255 )).$(( ip & 255 ))"
}
# Vérifier si le fichier des réservations existe
if [[ ! -f "$RESERVATION_FILE" ]]; then
echo "Erreur : Le fichier des réservations $RESERVATION_FILE n'existe pas."
exit 1
fi
# Lire toutes les IP utilisées dans le fichier de réservation
USED_IPS=($(grep -oE 'address=/[^/]+/[0-9]+\.[0-9]+\.[0-9]+\.[0-9]+' "$RESERVATION_FILE" | awk -F'/' '{print $3}'))
# Si aucune IP n'est réservée, retourner la première adresse disponible
if [[ ${#USED_IPS[@]} -eq 0 ]]; then
echo "✅ Première IP disponible : $IP_RANGE_START"
exit 0
fi
# Trouver la plus grande IP utilisée
MAX_IP_INT=0
for ip in "${USED_IPS[@]}"; do
ip_int=$(ip_to_int "$ip")
if (( ip_int > MAX_IP_INT )); then
MAX_IP_INT=$ip_int
fi
done
# Calculer la prochaine IP disponible
NEXT_IP_INT=$((MAX_IP_INT + 1))
NEXT_IP=$(int_to_ip "$NEXT_IP_INT")
# Vérifier que cette IP est bien dans la plage autorisée
START_INT=$(ip_to_int "$IP_RANGE_START")
END_INT=$(ip_to_int "$IP_RANGE_END")
if (( NEXT_IP_INT > END_INT )); then
echo "❌ Plus aucune IP disponible dans la plage définie."
exit 1
fi
echo "✅ Première IP disponible : $NEXT_IP"
exit 0