Dobles de tests y clases finales

por Fran Iglesias

Las clases marcadas como final indican que no pueden ser extendidas. Eso añade un extra de dificultad en ciertos tests.

En general, marcar las clases como finales es una buena práctica. Tanto es así, que una recomendación sería incluirlo en las plantillas y generadores de código de tu IDE. De este modo, tienes que decidir si necesitas quitarla y te ayudará a no usar la herencia cuando no es apropiada. Por defecto, usa final.

Pero si una clase tiene la marca final también significa que no se puede doblar en caso de que necesites pasarla como dependencia a una clase que estás testeando. Esto no es razón suficiente para dejar de marcar clases como finales, sino que debería servir como piedra de toque para reflexionar acerca de qué y cómo estamos testeando. Tampoco sería una buena idea usar Bypass-finals, una librería que elimina la clave al vuelo, permitiendo doblar esas clases.

Entonces, ¿cómo puedo usar clases finales como dobles?

Newables

La primera pregunta que yo me haría es la siguiente: ¿se trata de una clase newable?

Una clase newable es una clase que se pueda instanciar en cualquier momento y no tiene dependencias que no sean newables. En esta categoría entran Value Objects, Entidades, DTO, Eventos y otros objetos-mensaje, así como servicios que no tengan dependencias externas.

Los newables no necesitas doblarlos, puedes usarlos tal cual. Incluso en el nivel unitario.

Injectables

Las clases injectables son las que no podemos instanciar en cualquier momento y que, habitualmente, tienen dependencias externas, que se les pasan mediante inyección también cuando es necesario. También pueden ser dependencias de estado global o de servicios externos. Típicamente son las que definimos en el contendor de inyección de dependencias.

¿Qué podemos hacer con ellas? Si no tienen ninguna dependencia serían como newables, por lo que no deberías doblarlas.

En caso de tener dependencias, una opción que puedes considerar es tratar de doblar esas dependencias. Un repositorio que usa el EntityManager de Doctrine puede haber sido declarado como final, pero nada te impide instanciarlo y pasarle un doble del EntityManager. De este modo, doblas la dependencia externa y puedes seguir usando la clase final como colaboradora.

Usa la interfaz

Si la clase en cuestión implementa una interfaz o extiende una clase abstracta, crea el doble usando estas, ya sea mediante una librería o implementando clases anónimas ad hoc para el test.

Si la clase final no implementa una interfaz explícita, considera extraerla. No hace ningún daño y proporciona beneficios a futuro.

Si consideras que no tiene sentido extraer una interfaz, es posible que entonces tampoco tenga sentido tener la clase marcada como final. Como dice el artículo de Marco Pivetta, usar final no tiene sentido si no se cumplen dos condiciones:

  • No hay una abstracción (interfaz) que la clase final implemente.
  • Toda la API pública de la clase final es parte de la interfaz

Que básicamente se resumen en cumplir el principio de sustitución de Liskov (aplicado a extensión mediante herencia y las interfaces).

En ese caso, el de que usar final no tenga sentido, es mejor que desmarques la clase.

April 29, 2021

Etiquetas: testing   tdd  

Temas

good-practices

refactoring

php

testing

tdd

python

blogtober19

design-principles

design-patterns

tb-list

misc

bdd

legacy

golang

dungeon

ruby

tools

tips

hexagonal

ddd

bbdd

soft-skills

books

oop

javascript

api

sql

ethics

typescript

swift

java

agile