Vazamento de prefixos na mitigação DDoS
Objetivo
Ajudar no troubleshooting de um dos principais problemas relacionados com o não funcionamento de uma Mitigação DDoS, o vazamento dos prefixos /24. Quando se tem contratado uma nuvem de mitigação DDoS, precisamos garantir que o prefixo /24 atacado passe pelo Scrubbing Center da fornecedora de mitigação DDoS. Se o tráfego estiver vazando por outros trânsitos IP, IX, etc ou se o prefixo IP não tiver sendo devidamente anunciado, sua mitigação falhará e o ataque poderá vir por outros caminhos. Embora existam ataques em IPv6, o IPv4 ainda é o principal alvo dos ataques DDoS e por isso esse artigo está focado no IPv4. Provedores que já utilizam IPv6 se beneficiam de menor consumo de clean pipe, diminuição de efeitos colaterais causados pela mitigação e economia em CGNAT.
A maneira mais simples de checarmos os vazamos de prefixos /24, é usando os diversos Looking Glass que encontramos pela Internet. Neles podemos ver se os prefixos estão realmente passando por onde devem. Pensando em ajudar nessa tarefa criei um script em python que checa em alguns Looking Glass, como está o anúncio do prefixo e te avisar onde está o problema. O script é simples, não é tão performático porque os sistemas que são consultados também não são rápidos, mas acredito que deve ajudar mesmo assim ou até mesmo servir como prova de conceito para quem está estudando Redes.
O script lgview.py
Para utilizar o script você precisará do módulo netmiko instalado. Esse módulo te permite fazer conexões, via python, em diversos tipos de devices. Como utilizaremos o Debian aqui, é bem simples de instalar:
# apt install python3-netmiko
Instalado o netmiko basta criar o script abaixo. Este script é a primeira versão, pode conter bugs mas sinta-se convidado a ajudar a melhorá-lo. Pode me contactar para isso.
#!/usr/bin/env python3 ''' lgview is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. # This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. # You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA Precisa do pacote: python3-netmiko ''' import os import sys import re from netmiko import ConnectHandler __author__ = 'Marcelo Gondim' __version__= 1.0 __datebegin__= "25/06/2023" ######################################################################### routeviews = { "device_type": "cisco_xe", "host": "route-views.routeviews.org", "username": "rviews", "password": "" } ix_sp = { "device_type": "cisco_ios_telnet", "host": "lg.sp.ptt.br", } ix_rj = { "device_type": "cisco_ios_telnet", "host": "lg.rj.ix.br", } ix_ce = { "device_type": "cisco_ios_telnet", "host": "lg.ce.ix.br", } at_t = { "device_type": "cisco_ios_telnet", "host": "route-server.ip.att.net", "username": "rviews", "password": "rviews" } internexa = { "device_type": "juniper_junos", "host": "177.84.161.226", "username": "bgp_view", "password": "bgp_view" } algar = { "device_type": "juniper_junos", "host": "201.48.0.2", "username": "rviews", "password": "rviews" } ######################################################################### try: prefixo = sys.argv[1] asn_mitigacao = sys.argv[2] asn_prefixo = sys.argv[3] except: print("Parametros faltando! Precisa passar o prefixo IPv4 /24 mitigado, o ASN mitigador e o ASN do prefixo mitigado.") print("Ex.: ./lgview.py 192.168.0.0/24 65000 65001") exit(0) def grep(texto, dado): linhas = texto.split('\n') lista = [] for linha in linhas: if re.search(dado, linha): if "community" not in linha.lower(): lista.append(linha) return lista os.system('cls' if os.name == 'nt' else 'clear') titulo = "LGVIEW - Lista prefixos nos Looking Glass para troubleshooting de Mitigacao DDoS - %s - v%s - %s" % (__author__, __version__,__datebegin__) print("#"*126) print(" %s" %(titulo)) print("#"*126) print("Verificando Prefixo: " + prefixo + "\n") ######################################################################### print("Checando LG: route-views.routeviews.org") consulta = ConnectHandler(**routeviews) consulta.send_command('terminal length 0') resultado = consulta.send_command('show ip bgp ' + prefixo) busca = grep(resultado, asn_prefixo) for elemento in busca: if asn_mitigacao not in elemento: print('AS-PATH: ' + '\033[91m' + elemento + '\033[0m') else: print('AS-PATH: ' + '\033[92m' + elemento + '\033[0m') ######################################################################### print("Checando LG: IX-SP") consulta = ConnectHandler(**ix_sp) consulta.send_command('terminal length 0') resultado = consulta.send_command('show ip bgp ' + prefixo) busca = grep(resultado, asn_prefixo) for elemento in busca: if asn_mitigacao not in elemento: print('AS-PATH: ' + '\033[91m' + elemento + '\033[0m') else: print('AS-PATH: ' + '\033[92m' + elemento + '\033[0m') ######################################################################### print("Checando LG: IX-RJ") consulta = ConnectHandler(**ix_rj) consulta.send_command('terminal length 0') resultado = consulta.send_command('show ip bgp ' + prefixo) busca = grep(resultado, asn_prefixo) for elemento in busca: if asn_mitigacao not in elemento: print('AS-PATH: ' + '\033[91m' + elemento + '\033[0m') else: print('AS-PATH: ' + '\033[92m' + elemento + '\033[0m') ######################################################################### print("Checando LG: IX-CE") consulta = ConnectHandler(**ix_ce) consulta.send_command('terminal length 0') resultado = consulta.send_command('show ip bgp ' + prefixo) busca = grep(resultado, asn_prefixo) for elemento in busca: if asn_mitigacao not in elemento: print('AS-PATH: ' + '\033[91m' + elemento + '\033[0m') else: print('AS-PATH: ' + '\033[92m' + elemento + '\033[0m') ######################################################################### print("Checando LG: AT&T") consulta = ConnectHandler(**at_t) consulta.send_command('set cli screen-length 0') resultado = consulta.send_command('show route protocol bgp ' + prefixo) busca = grep(resultado, asn_prefixo) for elemento in busca: if asn_mitigacao not in elemento: print('AS-PATH: ' + '\033[91m' + elemento + '\033[0m') else: print('AS-PATH: ' + '\033[92m' + elemento + '\033[0m') ######################################################################### print("Checando LG: Internexa") consulta = ConnectHandler(**internexa) consulta.send_command('set cli screen-length 0') resultado = consulta.send_command('show route protocol bgp ' + prefixo) busca = grep(resultado, asn_prefixo) for elemento in busca: if asn_mitigacao not in elemento: print('AS-PATH: ' + '\033[91m' + elemento + '\033[0m') else: print('AS-PATH: ' + '\033[92m' + elemento + '\033[0m') ######################################################################### print("Checando LG: Algar") consulta = ConnectHandler(**algar) consulta.send_command('set cli screen-length 0') resultado = consulta.send_command('show route protocol bgp ' + prefixo) busca = grep(resultado, asn_prefixo) for elemento in busca: if asn_mitigacao not in elemento: print('AS-PATH: ' + '\033[91m' + elemento + '\033[0m') else: print('AS-PATH: ' + '\033[92m' + elemento + '\033[0m') #########################################################################
Para executar é bem simples, passe como parâmetros: o prefixo IPv4 /24 que quer fazer a busca, o ASN da empresa mitigadora e o ASN do seu prefixo. Abaixo um exemplo de chamada:
# ./lgview.py 198.51.100.0/24 65200 65001