Bez popisu

README-es.md 8.8KB

Clases - Sniffer Simple

Objetivos

  1. Practicar la declaración e implementación de clases en C++.

Pre-Lab:

Antes de llegar al laboratorio debes:

  1. Haber repasado la declaración e implementación de clases en C++.
  2. Haber estudiado los conceptos e instrucciones para la sesión de laboratorio.
  3. Haber tomado el quiz Pre-Lab que se encuentra en Moodle.

Sniffer Simple

Las computadoras se comunican con otras a través del protocolo de Internet (IP). Cuando una computadora envia información a otra computadora es via paquetes de Internet que contienen la dirección de Internet de la computadora que envia (computadora fuente), el puerto fuente de la aplicación que está enviando el mensaje, la dirección de Internet de la computadora que recibe (computadora destino), y el puerto de la aplicación que va a recibir el mensaje.

Podemos comparar las direcciones de Internet a las direcciones de una casa, y las aplicaciones a los miembros de una casa. Cuando se envia una carta de una casa a otra, la dirección en la carta identifica la casa destino, y el nombre en la carta identifica al miembro de la casa al que se le envía.

Por ejemplo cuando tu computadora de laboratorio esta contactactando al servidor de web del departamento, los paquetes que cargan la informacion de tu computadora al servidor de web contienen la dirección fuente de la computadora del laboratorio y la dirección destino del servidor de web; y el puerto fuente del tu buscador de web (browser) y el puerto destino del servidor de web.

Las direcciones de Internet son representados usando 4 bytes (32 bits), y comunmente son presentadas a los usuarios como cadenas de caracteres de 4 valores decimales. Cada valor decimal es la representacion decimal de uno de los 4 bytes: “(0-255).(0-255).(0-255).(0-255)”. Por ejemplo, las siguientes son tres direcciones IP 10.0.1.10, 192.168.10.11, 136.145.54.10.

Los números de los puertos son enteros sin signo en el rango de [0-65535]. Se representan usando 2 bytes (16 bits). Hay números de puertos que son asignados a servicios de aplicaciones comúnes tales como el número 22 para ssh, 23 para telnet, 25 smtp, y el 80 para http.

Para complicar las cosas un poco, cada tarjeta de red de computadoras tiene un identificador único que es usado para la comunicación entre tu computadora y el dispositivo de la red que enruta el tráfico de red de Internet y la red local a tu computadora y vice-versa (protocolo Ethernet). Este identificador único es conocido como la dirección de Hardware (también conocido como dirección MAC), es representado usando 6 bytes (48 bits), y es presentado a los usarios como una cadena de caracteres de 6 pares de dígitos hexadecimales (cada par de dígitos hexadecimal corresponde a 1 byte). Por ejemplo, los siguientes son direcciones MAC: e0:f8:47:01:e9:90 y 70:ad:60:ff:fe:dd:79:d8.

Un sniffer de paquetes (también conocido como analizador de paquetes, analizador de protocolos, o analizador de red) es un programa de computadora que puede interceptar y registrar tráfico pasando a través de una red digital, o dispositivo de red. Mientras los datos fluyen a través de la red, el sniffer captura cada paquet, y si es necesario decodifica los datos crudos del paquete[1].

Cada paquete capturado por este programa tiene la siguiente estructura:

  1. un encabezado de Ethernet que contiene las direcciones MAC fuente y destino
  2. un encabezado de IP que contiene las direcciones IP fuente y destino
  3. un encabezado que contiene los números de puerto fuente y destino

En esta experiencia de laboratorio completaremos un sniffer de paquetes sencillo que captura todos los paquetes de IP que fluyen a través de tu computadora de laboratorio, y alguna información adicional de los paquetes. Adicionalmente detecta las solicitudes no encriptadas de imagenes en la web, y despliega las imágenes en el GUI.


Figura 1 - La aplicación corriendo.


La Figura 1 muestra una foto de la aplicación. Cada fila en la tabla es la información de un paquete capturado, la caja de texto bajo la tabla presenta un resumen en ASCII del paquete seleccionado en la tabla, y la lista en el lado derecho presenta las imagenes que han sido solicitadas y se han visto en la tarjeta de red.

La aplicación que tu estas a punto de completar le da a los usuarios la habilidad de analizar el tráfico de red y monitorear imagenes que estan siendo vistas en tu red.

Sesión de laboratorio

Para crear un sniffer de paquetes puedes usar la librería de pcap que provee una interface para accesar la data que está pasando a través de tu tarjeta de red. Esta librería contiene una función que devuelve un torrente crudo de los bytes de cada paquete capturado.

Es la tarea del programador del sniffer decodificar el torrente en crudo a información legible por humanos. Afortunadamente esta no va a ser tu tarea, pero tu puedes aprender a hacerlo, si quieres, leyendo el código fuente de este laboratorio. Tu tarea es seguir los ejercicios abajo para que puedas proveerle al sniffer los objetos necesarios (Clases) para procesar los paquetes.

Ejercicio 1: Familiriarizate con la aplicación

Instrucciones

  1. Para cargar este proyecto necesitas correr qt creator con privilegios de administrador (root).

    ```sudo qtcreator Documents/eip/simplesniffer/SimpleSniffer.pro```
    
  2. El proyecto SimpleSniffer está en el directorio Documents/eip/simplesniffer de tu computadora. Alternativamente puedes crear un clon del repositorio git http://bitbucket.org/eip-uprrp/classes-simplesniffer para descargar el directorio classes-simplesniffer a tu computadora.

  3. Configura el proyecto. El proyecto consiste de varios archivos. En este laboratorio trabajarás con los archivos ethernet_hdr.h, ethernet_packet.h, ethernet_packet.cpp, ip_packet.h and ip_packet.cpp.

Ejercicio 2: Completa la clase ethernet_packet

Lee el archivo ethernet_hdr.h, este contiene la definición de la estructura de datos que representa un encabezado de Ethernet. A continuación mostramos esa definición:

#define ETHER_ADDR_LEN 6

struct sniff_ethernet {
        u_char  ether_dhost[ETHER_ADDR_LEN];    /* direccion destino */
        u_char  ether_shost[ETHER_ADDR_LEN];    /* direccion fuente */
        u_short ether_type;                     /* IP? ARP? RARP? etc */
};

El encabezado de Ethernet arriba es usado para decodificar la parte Ethernet de los datos crudos en cada paquete. Este se compone de la dirección MAC fuente (ether_shost, 6 bytes), la dirección MAC destino (ether_dhost, 6 bytes), y el tipo de paquete de Ethernet (ether_type, 2 bytes) que es usado para determinar si el paquete es un paquete de IP.

Como puedes ver, no es una buena idea enseñar este formato de información a un usario regular. Tu primer tarea es definir las funciones de la clase de C++ que define las funciones para traducir la información de las direcciones MAC a cadenas de caracteres legibles por humanos.

El encabezado de la clase está en el archivo ethernet_packet.h y también la mostramos a continuación:

class ethernet_packet
{

    sniff_ethernet ethernet ;
    // Devuelve una direccion de 6 bytes MAC en una cadena de caracteres.
    string mac2string(u_char []) ;

public:
    ethernet_packet();  // Constructor por defecto

    // Ajusta la variable miembro ether_host a los valores
    // recibidos en el arreglo
    void setEtherDHost(u_char []) ;
    // Lo mismo que arriba pero al  ether_shost
    void setEtherSHost(u_char []) ;

    // Ajusta el ethernet type al valor recibido.
    void setEtherType(u_short) ;

    // Devuelve la representación en cadenas de caracteres de las direcciones
    // Ethernet
    string getEtherDHost() ;
    string getEtherSHost() ;

    // Devuelve el tipo de ethernet.
    u_short getEtherType() ;

};

Implementa las funciones en el archivo ethetnet_packet.cpp.

[Rafa] Puedes sugerir una forma para que validen si han implementado bien la función.

Ejercicio 3: Construye el encabezado de ip_packet

Para el Ejercicio 3 debes estudiar las definiciones de las funciones de la clase ip_packet que se encuentra en el archivo ip_packet.cpp

Tu tarea es definir el encabezado de la clase siguiendo las funciones en ese archivo. Las variables miembro son:

  • dos variables string para almacenar las direcciones de IP fuente y destino
  • una variable de un byte (char) para almacenar el typo de protocolo IP
  • dos variables unsigned short para almacenar el puerto fuente y destino
  • una variable string para almacenar la carga del paquete.

En la declaración de la clase ip_packet debes especificar que es una clase derivada de la clase ethernet_packet.

Entregas

Utiliza “Entrega” en Moodle para entregar los archivos ethernet_packet.cpp y ip_packet.h que definistes.

Referencias

[1]http://en.wikipedia.org/wiki/Packet_analyzer