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

  • xBrowserSync in Docker

    29.03.2021

    Nachdem ich schon längere Zeit nicht mehr über neue Dienste in meinem Docker-Zoo berichtet habe, habe ich in der vergangenen Woche wieder einmal einen Neuzugang begrüßen dürfen...

    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
  • Windows? Nur noch gegen Bezahlung!

    Ich habe mich nun völlig von Windows - der armseligen Ausrede für ein Computerbetriebssystem aus Redmond - abgenabelt

    Weiterlesen
  • Vergleich Analoger und Digitaler Identitäten

    Eine Präsentation zum besseren Verständnis der Konzepte hinter digitalen Identitäten

    Aktualisierung vom 16. März 2025

    Aktualisierung der Präsentation mit einem Beispiel aus einem Film der 1980er Jahre und Betonung des Fakts, dass das Subjekt überhaupt nicht bemerken muss, dass eine Identität erstellt wird...

    Aktualisierung vom 17. August 2025

    Ein weiteres Beispiel wurde hinzugefügt.

    Aktualisierung vom 30. März 2026

    Aktualisierung der Präsentation: Erläuterung der Möglichkeit, mehr als ein Zertifikat für dasselbe Schlüsselpaar auszustellen und Exkurs zu Transport Layer Security als Beispiel der Forderung des Vorweisens bestimmter Arten digitaler Identitäten.
    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.