Tests mit Zeiten und Uhren in Java

vorhergehende Artikel in: Java Software-Test Komponenten
11.01.2023

Ich habe vor nicht allzu langer Zeit hier bereits darauf hingewiesen, dass vieles, das aktuell in riesigen Frameworks vermeintlich erfunden wurde und wird bereits (seit langem) Teil von Java SE ist.

Will man Tests - speziell Unit-Tests - in Java schreiben, bei denen man bestimmte Assertions auf die verronnene Zeit anwenden oder das Verhalten des eigenen Codes nach Ablauf einer ganz bestimmten Zeitspanne anwenden möchte, hat man das Problem, das heutigen Multicore-Architekturen inhärent ist: Man kann (ohne Echtzeitbetriebssystem) keine genauen Aussagen darüber machen, wie lange ein bestimmtes Stück Code für seine Abarbeitung benötigt.

Dann hat man in den oben beschriebenen Szenarien nur die Möglichkeit, relativ weite Grenzen für die Zeiten zu setzen, innerhalb derer der jeweilige Test noch als verifiziert gelten soll. Andererseits wird in der heutigen Softwareentwicklung der Code nicht nur einmal gebaut - in einer perfekten Welt sollte ein Build-Server existieren, der jeden Push auf bestimmte Branches mit einem Sanity-Check-Build beantwortet. Auf diesem Buildserver können aber ganz andere Ressourcenvoraussetzungen bestehen, als auf Entwicklerrechnern, was dazu führen kann, dass solche wie die beschriebenen Tests wieder reihenweise durchfallen.

Früher konnte man in Tests noch etwas erreichen, indem man statische Methoden zur Bestimmung der aktuellen Zeit gemockt hat. Heute gibt es einen eleganteren Weg: Man legt - zum Beispiel über entsprechende Architekturregeln, die mit korrespondierenden Tests automatisch überprüft werden - fest, dass nur noch die Methoden im Paket java.time für jegliche Operationen zur Bestimmung von Zeitpunkten benutzt werden. Diese Methoden akzeptieren alle einen (optionalen) Parameter vom Typ java.time.Clock - und die hierfür verwendete Instanz wird dann nicht direkt erzeugt, sondern von einem Dienst erfragt. In Tests kann man dann einen anderen Dienst benutzen, der eine MutableClock zurückliefert, deren Zeitwert sich beliebig manipulieren lässt. So arbeit zu testender Code immer mit der korrekten und wiederholt derselben Zeit.

Dieses System habe ich selber ausprobiert und dabei gleich noch eine Verbesserung vorgeschlagen: Normalerweise lassen sich BeanContext-Services ja nur von Beans Nutzen, die im gleichen BeanContext registriert sind, wie die Services. Das lässt sich durch die Verwendung zum Beispiel des folgenden Ansatzes aushebeln:

import java.beans.beancontext.BeanContextChildSupport;
import java.beans.beancontext.BeanContextServices;
import java.beans.beancontext.BeanContextServicesSupport;
import java.util.TooManyListenersException;

/** * Usable for testing. For example: * <pre> * MutableClock mc=MutableClock.of(Instant.ofEpochMilli(1000000000), ZoneId.systemDefault()); * GlobalContext.getContextServices().addService(ClockSource.class,new ClockSourceServiceProvider(mc)); * ClockSource clockSource= GlobalContext.getService(ClockSource.class); * System.out.println(Instant.now(clockSource.getClock())); * mc.add(5, ChronoUnit.MINUTES); * System.out.println(Instant.now(clockSource.getClock())); * </pre> */ public class GlobalContext { private static final BeanContextServicesSupport context = new BeanContextServicesSupport(); // a bean context private static final BeanContextChildSupport bean=new BeanContextChildSupport(); static { getContextServices().add(bean); }

public static BeanContextServices getContextServices() { return context; }

public static <T> T getService(Class<T> cls) throws TooManyListenersException { return (T)getContextServices().getService(bean,bean,cls,bean,bean); } }

Ein angenehmer Nebeneffekt dabei ist, dass die zurückgegebene Instanz durch die Generifizierung der Methode nicht - wie sonst im Umfeld des BeanContext allgemein üblich - erst noch gecastet werden muss, sondern direkt verwendet werden kann.

Alle Artikel rss Wochenübersicht Monatsübersicht Codeberg Repositories Mastodon Über mich home xmpp


Vor 5 Jahren hier im Blog

  • Derangement - theoretische Betrachtungen

    23.12.2020

    Ich habe bereits über meine Implementierung eines Derangement berichtet - hier noch einige theoretische Nachbetrachtungen...

    Weiterlesen

Neueste Artikel

  • Asymmetrische Kryptographie

    Ich habe mich mit der Idee schon länger getragen: Nochmal einen Rundumschlag zu asymmetrischer Kryptographie zu machen. Dabei werde ich mich auf Demonstrationen der einzelnen Konzepte und Operationen mit Beispielcode konzentrieren und zu jedem der vorgestellten Konzepte mehr oder weniger ausführlich bezüglich der Einsatzszenarien und Vor- und Nachteile Stellung beziehen

    Weiterlesen
  • Eingereichter Vortag zum 39C3 - Gentlemen - check your architecture!

    Dieser Vortrag wurde zum 39C3 eingereicht und abgelehnt. Ich möchte einige Open-Source-Frameworks für verschiedene Programmiersprachen vorstellen, mit denen sich Architekturregeln pro Projekt oder Organisation festlegen und mithilfe gängiger Testinfrastrukturen durchsetzen lassen.

    Weiterlesen
  • Chatkontrolle vermeiden ist gleichzeitig unwichtig und nicht genug!

    Das ist ein Abstract eines Vortrages, den ich auf dem 39C3 halten wurde, der aber abgelehnt wurde

    Weiterlesen

Manche nennen es Blog, manche Web-Seite - ich schreibe hier hin und wieder über meine Erlebnisse, Rückschläge und Erleuchtungen bei meinen Hobbies.

Wer daran teilhaben und eventuell sogar davon profitieren möchte, muss damit leben, daß ich hin und wieder kleine Ausflüge in Bereiche mache, die nichts mit IT, Administration oder Softwareentwicklung zu tun haben.

Ich wünsche allen Lesern viel Spaß und hin und wieder einen kleinen AHA!-Effekt...

PS: Meine öffentlichen Codeberg-Repositories findet man hier.