TLSA-Record überprüfen

Du hast DNSSEC erfolgreich installiert und möchtest regelmäßig deinen TLSA-Record für die Mailzustellung prüfen? Gerade bei der Verwendung von Let’s Encrypt Zertifikaten ändern sich die HASH-Werte im DNS. Wenn du das nicht beachtest oder es nicht mitbekommst, wird die Absicherung deines Mailservers mit DANE ungültig. Das wollen wir natürlich vermeiden.

Also erstellen wir ein Script, das uns informiert, wenn es Abweichungen gibt. Ich speichere das Script in /opt/scripts/dnssec und nenne es tlsa.sh.

#!/bin/bash

# Usage: ./tlsa.sh <hostname> <port> <protocol> [<nameserver>]
# Example: ./tlsa.sh mx.mein-mail-server.de 25 tcp 8.8.8.8

if [ "$#" -lt 3 ] || [ "$#" -gt 4 ]; then
    echo "Usage: $0 <hostname> <port> <protocol> [<nameserver>]"
    exit 1
fi

HOSTNAME=$1
PORT=$2
PROTO=$3
NAMESERVER=$4
EMAIL="your-email@example.com"  # Set your email address here

# Check if required tools are installed
for cmd in openssl dig xxd mail; do
    if ! command -v $cmd &>/dev/null; then
        echo "Error: '$cmd' is not installed. Install it and try again."
        exit 1
    fi
done

# Format the hostname for TLSA query
TLSA_HOSTNAME="_${PORT}._${PROTO}.${HOSTNAME}"

# Get the TLSA record from DNS (with optional NS)
if [ -n "$NAMESERVER" ]; then
    TLSA_RECORD=$(dig +short @${NAMESERVER} TLSA $TLSA_HOSTNAME | awk '{print $4 $5 $6 $7}')
else
    TLSA_RECORD=$(dig +short TLSA $TLSA_HOSTNAME | awk '{print $4 $5 $6 $7}')
fi

if [ -z "$TLSA_RECORD" ]; then
    echo "Error: No TLSA record found for $TLSA_HOSTNAME"
    exit 1
fi

echo "TLSA-Record: $TLSA_RECORD"

# Retrieve the certificate from the server and calculate the public key hash
CALCULATED_HASH=$(echo | openssl s_client -connect $HOSTNAME:$PORT -starttls smtp -showcerts 2>/dev/null | \
    awk '/-----BEGIN CERTIFICATE-----/,/-----END CERTIFICATE-----/' | \
    openssl x509 -pubkey -noout | openssl pkey -pubin -outform DER | \
    openssl dgst -sha256 -binary | xxd -p -c 32)

echo "Calculated Hash: $CALCULATED_HASH"

# Convert both hashes to lowercase for comparison
TLSA_RECORD_LOWER=$(echo "$TLSA_RECORD" | tr '[:upper:]' '[:lower:]')
CALCULATED_HASH_LOWER=$(echo "$CALCULATED_HASH" | tr '[:upper:]' '[:lower:]')

# Compare calculated hash with TLSA record
if [ "$CALCULATED_HASH_LOWER" == "$TLSA_RECORD_LOWER" ]; then
    echo "✅ TLSA record matches the certificate!"
else
    echo "❌ TLSA record does NOT match the certificate!"
    echo -e "Subject: TLSA Record Mismatch\n\nThe TLSA record for $HOSTNAME on port $PORT ($PROTO) does not match the calculated hash." | mail -s "TLSA Record Mismatch" $EMAIL
fi

Anschließend lassen wir das Script relgelmäßig per cron laufen

crontab -e
###
0 4 * * * /opt/scripts/dnssec/tlsa.sh