Cuando trabajas en una función que depende del tiempo ... ¿Cómo organizas las pruebas unitarias? Cuando los escenarios de las pruebas unitarias dependen de la forma en que su programa interpreta "ahora", ¿cómo los configura?
Segunda edición: después de un par de días leyendo tu experiencia
Puedo ver que las técnicas para lidiar con esta situación generalmente giran en torno a uno de estos tres principios:
- Agregue (código duro) una dependencia: agregue una pequeña capa sobre la función / objeto de tiempo, y siempre llame a su función de fecha y hora a través de esta capa. De esta manera, puede obtener el control del tiempo durante los casos de prueba.
- Usa simulacros: tu código permanece exactamente igual. En sus pruebas, reemplaza el objeto de tiempo por un objeto de tiempo falso. A veces, la solución implica modificar el objeto de tiempo genuino que proporciona su lenguaje de programación.
- Use la inyección de dependencia: cree su código para que cualquier referencia de tiempo se pase como parámetro. Luego, tiene el control de los parámetros durante las pruebas.
Las técnicas (o bibliotecas) específicas de un idioma son muy bienvenidas, y se mejorarán si aparece un código ilustrativo. Entonces, lo más interesante es cómo se pueden aplicar principios similares en cualquier plataforma. Y sí ... si puedo aplicarlo de inmediato en PHP, mejor que mejor;)
Comencemos con un ejemplo simple: una aplicación de reserva básica.
Digamos que tenemos API JSON y dos mensajes: un mensaje de solicitud y un mensaje de confirmación. Un escenario estándar es el siguiente:
- Usted hace una solicitud Obtienes una respuesta con un token. El sistema bloquea el recurso que se necesita para cumplir esa solicitud durante 5 minutos.
- Confirma una solicitud, identificada por el token. Si el token se emitió dentro de los 5 minutos, se aceptará (el recurso aún está disponible). Si han pasado más de 5 minutos, debe realizar una nueva solicitud (el recurso fue liberado. Debe verificar su disponibilidad nuevamente).
Aquí viene el escenario de prueba correspondiente:
Hago una solicitud Confirmo (inmediatamente) con el token que recibí. Mi confirmación es aceptada
Hago una solicitud Espero 3 minutos Confirmo con el token que recibí. Mi confirmación es aceptada
Hago una solicitud Espero 6 minutos Confirmo con el token que recibí. Mi confirmación es rechazada.
¿Cómo podemos llegar a programar estas pruebas unitarias? ¿Qué arquitectura debemos usar para que estas funcionalidades sigan siendo comprobables?
Nota de edición : cuando disparamos una solicitud, el tiempo se almacena en una base de datos en un formato que pierde cualquier información sobre milisegundos.
EXTRA - pero quizás un poco detallado: Aquí vienen detalles sobre lo que descubrí haciendo mi "tarea":
Construí mi función con una dependencia de una función de tiempo propia. Mi función VirtualDateTime tiene un método estático get_time () al que llamo donde solía llamar al nuevo DateTime (). Esta función de tiempo me permite simular y controlar qué hora es "ahora", de modo que pueda crear pruebas como: "Configurar ahora para el 21 de enero de 2014 16h15. Hacer una solicitud. Avanzar 3 minutos. Hacer la confirmación". Esto funciona bien, a costa de la dependencia, y "código no tan bonito".
Una solución un poco más integrada sería construir una función myDateTime propia que extienda DateTime con funcionalidades adicionales de "tiempo virtual" (lo más importante, ahora configúrelo como quiero). Esto está haciendo que el código de la primera solución sea un poco más elegante (uso del nuevo myDateTime en lugar del nuevo DateTime), pero termina siendo muy similar: tengo que construir mi característica usando mi propia clase, creando así una dependencia.
Pensé en hackear la función DateTime, para que funcione con mi dependencia cuando la necesito. Sin embargo, ¿hay alguna forma simple y ordenada de reemplazar una clase por otra? (Creo que obtuve una respuesta a eso: ver el espacio de nombres a continuación).
En un entorno PHP, leí que "Runkit" puede permitirme hacer esto (piratear la función DateTime) dinámicamente. Lo que es bueno es que podría verificar que estoy corriendo en un entorno de prueba antes de modificar cualquier cosa sobre DateTime, y dejarlo intacto en producción [1] . Esto suena mucho más seguro y limpio que cualquier pirateo manual de DateTime.
Inyección de dependencia de una función de reloj en cada clase que utiliza el tiempo [2] . ¿No es esto excesivo? Veo que en algunos entornos se vuelve muy útil [5] . Sin embargo, en este caso, no me gusta tanto.
Eliminar la dependencia del tiempo en todas las funciones [2] . ¿Es esto siempre factible? (Ver más ejemplos a continuación)
¿Usando espacios de nombres [2] [3] ? Esto se ve bastante bien. Podría burlarme de DateTime con mi propia función DateTime que extiende \ DateTime ... Use DateTime :: setNow ("2014-01-21 16:15:00") y DateTime :: wait ("+ 3 minutos"). Esto cubre más o menos lo que necesito. ¿Qué pasa si usamos la función time ()? ¿O otras funciones de tiempo? Todavía tengo que evitar su uso en mi código original. O tendría que asegurarme de que cualquier función de tiempo PHP que utilizo en mi código se anule en mis pruebas ... ¿Hay alguna biblioteca disponible que haga exactamente esto?
He estado buscando una manera de cambiar la hora del sistema solo por un hilo. Parece que no hay ninguno [4] . Es una pena: una función PHP "simple" para "Establecer el tiempo hasta el 21 de enero de 2014 16h15 para este hilo" sería una gran característica para este tipo de pruebas.
Cambie la fecha y hora del sistema para la prueba, usando exec (). Esto puede funcionar si no tiene miedo de meterse con otras cosas en el servidor. Y debe retrasar la hora del sistema después de ejecutar su prueba. Puede hacer el truco en algunas situaciones, pero se siente bastante "hacky".
Esto me parece un problema muy estándar. Sin embargo, todavía extraño una forma simple y genérica de tratarlo. Tal vez me perdí algo? ¡Por favor comparte tu experiencia!
NOTA: Aquí hay otras situaciones en las que podemos tener necesidades de prueba similares.
Cualquier característica que funcione con algún tipo de tiempo de espera (por ejemplo, ¿juego de ajedrez?)
procesando una cola que desencadena eventos en un momento dado (en el escenario anterior podríamos seguir así: un día antes de que comience la reserva, quiero enviar un correo al usuario con todos los detalles. Escenario de prueba ...)
Desea configurar un entorno de prueba con datos recopilados en el pasado, le gustaría verlo como si fuera ahora. [1]
1 /programming/3271735/simulate-different-server-datetimes-in-php
2 /programming/4221480/how-to-change-current-time-for-unit-testing-date-functions-in-php
3 http://www.schmengler-se.de/en/2011/03/php-mocking-built-in-functions-like-time-in-unit-tests/
DateTime now
al código en lugar de un reloj.