Utiliza este crédito para lanzar tus proyectos ahora en Digital Ocean, válido por 60 días
Índice del contenido
Go: testing básico y coverage
Go: testing básico y coverage
Go ya cuenta con un modulo de testing en su librería estándar que está lista para nuestro uso, solo hace falta importarlo y usarlo.
Preparación del testing en go
Para que se lleven a cabo los tests necesitamos:
- Un archivo que termine en _test.go
- Correr el comando go test
├── go.mod
├── main.go
└── testing
├── main.go
└── main_test.go
1 directory, 5 files
Considera que, si vas a asignarle un nombre a tu paquete , jamás debeser nombrarlo testing. ¿Por qué? si lo haces, go confungirá su paquete testing con el tuyo, devolviéndote esos resultados incorrectos.
Para crear los tests, dentro del archivo testing/main_test.go, necesitamos una función que reciba de argumento nuestro paquete de testing con el carácter de desestructuración.
Compararemos el resultado usando un if, o lo que querramos y, si el test falla, llamaremos al método Errorf del modulo de testing.
package main
import "testing"
func TestDivision(t *testing.T) {
total := Division(10, 2)
if total != 5 {
t.Errorf("División incorrecta, obtuvimos %d pero se esperaba %d", total, 5)
}
}
No es necesario que las funciones a probar se encuentren dentro del archivo de testing, en este caso las coloqué dentro de testing/main.go.
package main
func Division(a int, b int) int {
return a / b
}
¿Por qué no ir más allá de los fundamentos de Go hacia algo más avanzado?
PatrocinadoEl otro día encontré este curso de Go, trata sobre el nucleo de Go como lenguaje, así como conceptos avanzados como manejo de errores y redes, técnicas de programación eficientes y errores comunes de este incomprendido lenguaje tan potente.
Ejecutar los tests
Para ejecutar los tests necesitamos encontrarnos dentro del directorio donde se encuentran nuestros archivos terminados en _test.go y correr el comando go test. Si el test aprueba obtendremos la leyenda PASS.
cd testing/
go test
PASS
ok main/testing 0.001s
Por otro lado, si los tests fallan se imprimirá la palabra FAIL en pantalla:
--- FAIL: TestDivision (0.00s)
main_test.go:14: División incorrecta, obtuvimos 12 pero se esperaba 5
FAIL
exit status 1
FAIL main/testing 0.001s
Manejo de casos con tablas
En el ejemplo anterior usamos una función para probar un caso, sin embargo, si necesitaramos someter a prueba múltiples casos, necesitariamos una función por cada test, bastante tedioso, ¿no?
Para evitar llenarse de funciones, los desarrolladores usan un array compuesto de structs, donde cada struct representa un caso a probar. Puedes pensar en el array de structs como una tabla, donde cada fila es un caso y cada columna un tipo de dato a probar.
En este caso, cada struct de nuestro array consiste en tres enteros; los primeros dos representan los argumentos, mientras que el último es el resultado.
tables := []struct {
x int
y int
r int
}{
{100, 10, 10}, // 100 / 10 = 10
{200, 20, 10}, // 200 / 20 = 10
{300, 30, 10},
{1000, 100, 10},
}
De seguro ya notaste que no estamos cubriendo la división entre cero, pero déjalo así por ahora.
Ya que contamos con nuestro array de structs, iteraremos sobre cada uno de sus elementos usando la función range de go . De esta manera cubriremos cada caso.
for _, table := range tables {
total := Division(table.x, table.y)
if total != table.r {
t.Errorf("División de %d entre %d incorrecta, obtuvimos: %d, pero el resultado es: %d.", table.x, table.y, total, table.r)
}
}
Si todo salió bien, pasaremos todas las pruebas.
Coverage
Coverage ya forma parte del código en go, por lo que no necesitamos librerías externas. Si no sabes que es Coverage, piensa en él como el porcentaje de tu código que es sometido a pruebas. Si todo tú código pasa por las pruebas tendrás un coverage de 100%, si solo la mitad pasa por las pruebas el coverage será de 50%. Anteriormente te hablé del coverage en mi entrada unittest en Python.
Para calcular el coverage basta con agregar el flag -cover al comando go test
go test -cover
PASS
coverage: 100.0% of statements
ok _/home/eduardo/Programacion/goTesting/testing 0.002s
Como nuestra función es muy corta, obtenemos un resultado de 100%, sin desglosar, de coverage
Exportar resultados de coverage
Podemos mandar el toda la información en bruto de nuestro test de coverage a un archivo externo con el flag -coverprofile.
go test -coverprofile=coverage.out
mode: set
/home/eduardo/Programacion/goTesting/testing/main.go:3.33,5.2 1 1
Este archivo, de nombre coverage.out, que fue generado, es un archivo que contiene información en bruto y que será necesario para visualizar los resultados de una manera más detallada.
Visualización de resultados con go tool
Para resumir de una manera más legible la información del archivo que contiene nuestro test de coverage, usaremos el comando tool, acompañado del flag -func, seguido del nombre del archivo. Lo que nos devolverá un resultado de coverage desglosado.
go tool cover -func=coverage.out
/home/eduardo/Programacion/goTesting/testing/main.go:3: Division 100.0%
total: (statements) 100.0%
Go también nos permite visualizar el coverage en formato HTML, con colores, directo en nuestro navegador. Para ello usamos la opción -html, seguido del archivo con los datos de coverage.
Al ejecutar el comando, se abrirá una pestaña de nuestro navegador y nos mostrará los resultados testeados en verde y los no testeados en rojos.
go tool cover -html=coverage.out
Coverage completo en html en go
Si decidimos modificar nuestra función para que maneje los casos de la división por cero, y corremos los tests de coverage de nuevo, obtendremos un esquema diferente al anterior. Ahora sí aparece una sección de código no cubierto por los tests en colo rojo y nuestro coverage bajó a 50%.
Coverage incompleto en html en go
Con esto termino esta entrada super corta sobre testeo en go. Para la siguiente entrada voy a hablar un poquito de profiling y daré por terminadas las entradas de go básico para escribir nuevamente sobre Python.