feat: sshconnect — refactoring UX et robustesse

- Serveurs ON affichés en premier (tri ON>OFF puis alphabétique)
- Affichage user@hostname dans la liste
- Panneau preview fzf : hostname/user/port/identityfile par host
- --ansi pour les couleurs ON (vert) / OFF (rouge)
- Champ status caché (|ON|OFF|) pour logique propre sans regex sur ANSI
- exec ssh remplace le process shell (pas de fork inutile)
- Passage à -P 20 pour les parcs avec beaucoup de serveurs
- read -r et printf à la place de echo -e
This commit is contained in:
2026-04-30 08:39:59 +02:00
parent e6263c1ae2
commit 2b8690c34b
+39 -29
View File
@@ -1,50 +1,60 @@
#!/bin/bash #!/bin/bash
# 1. Liste des hosts mapfile -t ALL_HOSTS < <(
mapfile -t ALL_HOSTS < <(grep -riI "^Host " ~/.ssh/config ~/.ssh/config.d/ ~/.ssh/include/ 2>/dev/null | awk '{print $2}' | grep -v '*' | sort -u) grep -hriI "^Host " ~/.ssh/config ~/.ssh/config.d/ ~/.ssh/include/ 2>/dev/null \
| awk '{print $2}' \
| grep -v '\*' \
| sort -u
)
TOTAL=${#ALL_HOSTS[@]}
check_host() { check_host() {
local host=$1 local host=$1
local info=$(ssh -G "$host") local info addr port user
local addr=$(echo "$info" | awk '/^hostname / {print $2}') info=$(ssh -G "$host" 2>/dev/null)
local port=$(echo "$info" | awk '/^port / {print $2}') addr=$(awk '/^hostname / {print $2}' <<< "$info")
port=$(awk '/^port / {print $2}' <<< "$info")
user=$(awk '/^user / {print $2}' <<< "$info")
# Test de port TCP
if (timeout 0.7s bash -c "cat < /dev/null > /dev/tcp/$addr/$port") 2>/dev/null; then if (timeout 0.7s bash -c "cat < /dev/null > /dev/tcp/$addr/$port") 2>/dev/null; then
# On utilise "|" comme délimiteur interne printf "\033[32m[ ON ]\033[0m|ON|%-22s|%s@%s\n" "$host" "$user" "$addr"
printf "[ ON ]|%s\n" "$host"
else else
printf "[ OFF ]|%s\n" "$host" printf "\033[31m[ OFF ]\033[0m|OFF|%-22s|%s@%s\n" "$host" "$user" "$addr"
fi fi
} }
export -f check_host export -f check_host
echo "Vérification des serveurs..."
# 2. Scan parallèle printf "Vérification de %d serveurs...\n" "$TOTAL"
STATE_LIST=$(printf "%s\n" "${ALL_HOSTS[@]}" | xargs -I {} -P 10 bash -c 'check_host "{}"')
# 3. Interface FZF # Scan parallèle — ON affiché en premier, puis tri alphabétique
# On demande à fzf d'afficher les colonnes proprement STATE_LIST=$(
choice=$(echo "$STATE_LIST" | fzf --height 40% --reverse \ printf "%s\n" "${ALL_HOSTS[@]}" \
--delimiter="\|" \ | xargs -I {} -P 20 bash -c 'check_host "{}"' \
--with-nth=1,2 \ | sort -t'|' -k2,2r -k3,3
--header "Tapez 'OFF' pour les serveurs HS | 'ON' pour les actifs") )
choice=$(echo "$STATE_LIST" | fzf \
--ansi \
--height 70% \
--reverse \
--delimiter="|" \
--with-nth=1,3,4 \
--header "Entrée: connexion | ESC: annuler | Filtre: ON / OFF / nom" \
--preview 'host=$(echo {} | cut -d"|" -f3 | tr -d " "); ssh -G "$host" 2>/dev/null | grep -E "^(hostname|user|port|identityfile|compression) " | column -t' \
--preview-window=right:45%:wrap)
# 4. Connexion propre
if [ -n "$choice" ]; then if [ -n "$choice" ]; then
# On extrait le nom de l'hôte en utilisant le délimiteur "|" status=$(echo "$choice" | cut -d'|' -f2)
host_to_connect=$(echo "$choice" | cut -d'|' -f2) host_to_connect=$(echo "$choice" | cut -d'|' -f3 | tr -d '[:space:]')
# Nettoyage radical des caractères invisibles ou espaces restants if [[ "$status" == "ON" ]]; then
host_to_connect=$(echo "$host_to_connect" | tr -d '[:space:]')
if [[ "$choice" == *"[ ON ]"* ]]; then
clear clear
ssh "$host_to_connect" exec ssh "$host_to_connect"
else else
echo -e "\033[0;31m⚠ Le serveur $host_to_connect semble OFFLINE.\033[0m" printf "\033[0;31m⚠ Le serveur '%s' semble OFFLINE.\033[0m\n" "$host_to_connect"
read -p "Tenter quand même la connexion ? (y/n) " confirm read -rp "Tenter quand même la connexion ? (y/n) : " confirm
[[ $confirm == [yY] ]] && ssh "$host_to_connect" [[ $confirm == [yY] ]] && exec ssh "$host_to_connect"
fi fi
fi fi