From 355c43b2f3fb11f0a7a1eac55bd23dbd737eaf62 Mon Sep 17 00:00:00 2001 From: zlukich Date: Tue, 6 Feb 2024 13:28:38 +0100 Subject: [PATCH 01/12] started work on fingerprint comparator --- .../org/matsim/core/events/EventsUtils.java | 45 ++++++++++++++----- .../EventsFileFingerprintComparator.java | 25 +++++++++++ 2 files changed, 60 insertions(+), 10 deletions(-) create mode 100644 matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparator.java diff --git a/matsim/src/main/java/org/matsim/core/events/EventsUtils.java b/matsim/src/main/java/org/matsim/core/events/EventsUtils.java index 1f2ba7e3d33..c5bdc3f29d6 100644 --- a/matsim/src/main/java/org/matsim/core/events/EventsUtils.java +++ b/matsim/src/main/java/org/matsim/core/events/EventsUtils.java @@ -25,15 +25,17 @@ import org.matsim.core.config.Config; import org.matsim.core.controler.Injector; import org.matsim.utils.eventsfilecomparison.EventsFileComparator; +import org.matsim.utils.eventsfilecomparison.EventsFileFingerprintComparator; +import org.matsim.utils.eventsfilecomparison.FingerprintEventHandler; public final class EventsUtils { /** * Create a events manager instance that guarantees causality of processed events across all handlers. */ - public static EventsManager createEventsManager() { + public static EventsManager createEventsManager() { return new EventsManagerImpl(); - } + } /** * Creates a parallel events manager, with no guarantees for the order of processed events between multiple handlers. @@ -43,7 +45,7 @@ public static EventsManager createParallelEventsManager() { } public static EventsManager createEventsManager(Config config) { - final EventsManager events = Injector.createInjector( config, new EventsManagerModule() ).getInstance( EventsManager.class ); + final EventsManager events = Injector.createInjector(config, new EventsManagerModule()).getInstance(EventsManager.class); // events.initProcessing(); return events; } @@ -58,21 +60,44 @@ public static EventsManager getParallelFeedableInstance(EventsManager events) { return events; } else if (events instanceof ParallelEventsManager) { return events; - } - else if (events instanceof SynchronizedEventsManagerImpl) { + } else if (events instanceof SynchronizedEventsManagerImpl) { return events; } else { return new SynchronizedEventsManagerImpl(events); } } - public static void readEvents( EventsManager events, String filename ) { - new MatsimEventsReader(events).readFile(filename) ; + public static void createEventsFingerprint(String eventFile, String outputFingerprintFile) { + + EventsManager manager = createEventsManager(); + FingerprintEventHandler fingerprintEventHandler = new FingerprintEventHandler(); + manager.addHandler(fingerprintEventHandler); + + EventsUtils.readEvents(manager, eventFile); + + FingerprintEventHandler.writeEventFingerprintToFile(outputFingerprintFile,fingerprintEventHandler.eventFingerprint); + + } + + public static EventsFileComparator.Result createAndCompareEventsFingerprint(String inputFingerprint, String eventFile) { + + // header byte, version byte + // bin array time stamps + // event type counter map + // one hash (sha1?) + + new EventsFileFingerprintComparator(); + + return EventsFileComparator.Result.FILES_ARE_EQUAL; + } + + public static void readEvents(EventsManager events, String filename) { + new MatsimEventsReader(events).readFile(filename); } - public static EventsFileComparator.Result compareEventsFiles( String filename1, String filename2 ) { - EventsFileComparator.Result result = EventsFileComparator.compare( filename1, filename2 ); - return result ; + public static EventsFileComparator.Result compareEventsFiles(String filename1, String filename2) { + EventsFileComparator.Result result = EventsFileComparator.compare(filename1, filename2); + return result; } } diff --git a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparator.java b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparator.java new file mode 100644 index 00000000000..5e0bf7f606e --- /dev/null +++ b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparator.java @@ -0,0 +1,25 @@ +package org.matsim.utils.eventsfilecomparison; + +import it.unimi.dsi.fastutil.Hash; +import org.apache.logging.log4j.LogManager; +import org.apache.logging.log4j.Logger; +import org.matsim.api.core.v01.events.Event; +import org.matsim.core.events.handler.BasicEventHandler; + +import java.io.*; +import java.util.*; +import java.util.concurrent.CyclicBarrier; + +public class EventsFileFingerprintComparator { + + private static final Logger log = LogManager.getLogger(EventsFileComparator.class); + + public enum Result { FILES_ARE_EQUAL, DIFFERENT_NUMBER_OF_TIMESTEPS, DIFFERENT_TIMESTEPS, MISSING_EVENT, WRONG_EVENT_COUNT } + + public static EventsFileFingerprintComparator.Result compare(final String filename1, final String filename2) { + return null; + } + + +} + From 31917fe0b3b429f2a1ed82a64d2e1bd9c0e4cc9a Mon Sep 17 00:00:00 2001 From: zlukich Date: Wed, 21 Feb 2024 17:46:29 +0100 Subject: [PATCH 02/12] added compare function, fixed hashtoadd --- .../EventsFileFingerprintComparator.java | 42 ++++- .../FingerprintEventHandler.java | 177 ++++++++++++++++++ 2 files changed, 217 insertions(+), 2 deletions(-) create mode 100644 matsim/src/main/java/org/matsim/utils/eventsfilecomparison/FingerprintEventHandler.java diff --git a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparator.java b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparator.java index 5e0bf7f606e..94ba3b585a8 100644 --- a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparator.java +++ b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparator.java @@ -4,6 +4,9 @@ import org.apache.logging.log4j.LogManager; import org.apache.logging.log4j.Logger; import org.matsim.api.core.v01.events.Event; +import org.matsim.core.api.experimental.events.EventsManager; +import org.matsim.core.events.EventsManagerImpl; +import org.matsim.core.events.EventsUtils; import org.matsim.core.events.handler.BasicEventHandler; import java.io.*; @@ -16,8 +19,43 @@ public class EventsFileFingerprintComparator { public enum Result { FILES_ARE_EQUAL, DIFFERENT_NUMBER_OF_TIMESTEPS, DIFFERENT_TIMESTEPS, MISSING_EVENT, WRONG_EVENT_COUNT } - public static EventsFileFingerprintComparator.Result compare(final String filename1, final String filename2) { - return null; + public static Result compare(final String fingerprint, final String eventsfile) { + FingerprintEventHandler handler = new FingerprintEventHandler(); + EventsManager manager = EventsUtils.createEventsManager(); + + manager.addHandler(handler); + + EventsUtils.readEvents(manager,eventsfile); + + FingerprintEventHandler.EventFingerprint fingerprintFromEvents = handler.eventFingerprint; + + FingerprintEventHandler.EventFingerprint correctFingerprint = FingerprintEventHandler.readEventFingerprintFromFile(fingerprint); + + + if (fingerprintFromEvents == fingerprintFromEvents) { + return Result.FILES_ARE_EQUAL; + } + if (fingerprintFromEvents == null) { + return Result.DIFFERENT_TIMESTEPS; + } + + // Compare timeArray + if (!Objects.equals(fingerprintFromEvents.timeArray, fingerprintFromEvents.timeArray)) { + return Result.DIFFERENT_TIMESTEPS; + } + + // Compare eventTypeCounter + if (!Objects.equals(fingerprintFromEvents.eventTypeCounter, fingerprintFromEvents.eventTypeCounter)) { + return Result.WRONG_EVENT_COUNT; + } + + // Compare stringHash + if (!Objects.equals(fingerprintFromEvents.stringHash, fingerprintFromEvents.stringHash)) { + return Result.MISSING_EVENT; + } + + // All fields are equal + return Result.FILES_ARE_EQUAL; } diff --git a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/FingerprintEventHandler.java b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/FingerprintEventHandler.java new file mode 100644 index 00000000000..e65ce46bdc5 --- /dev/null +++ b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/FingerprintEventHandler.java @@ -0,0 +1,177 @@ +package org.matsim.utils.eventsfilecomparison; + +import it.unimi.dsi.fastutil.Hash; +import org.matsim.api.core.v01.events.Event; +import org.matsim.core.events.handler.BasicEventHandler; + +import java.io.*; +import java.security.MessageDigest; +import java.util.*; + +public class FingerprintEventHandler implements BasicEventHandler { + + public boolean ignoringCoordinates = false; + + public static byte header = 1; + public static byte version = 1; + public final EventFingerprint eventFingerprint = new EventFingerprint(); + + + public static class EventFingerprint { + List timeArray = new ArrayList<>(); + public Map eventTypeCounter = new HashMap<>(); + Integer stringHash; + + public EventFingerprint(List timeArray, Map eventTypeCounter, Integer stringHash) { + this.timeArray = timeArray; + this.eventTypeCounter = eventTypeCounter; + this.stringHash = stringHash; + } + + public EventFingerprint() { + this.timeArray = timeArray; + this.eventTypeCounter = eventTypeCounter; + Hash stringHash; + } + + public List getTimeArray() { + return timeArray; + } + + public Map getEventTypeCounter() { + return eventTypeCounter; + } + + public int getStringHash() { + return stringHash; + } + + public void addHashCode(String stringToAdd) { + if (stringToAdd == null) { + return; + } + + int hashToAdd = stringToAdd.hashCode(); + if (stringHash == null) { + stringHash = hashToAdd; + } else { + stringHash += hashToAdd; + } + } + } + + public void addEventType(String str) { + // Increment the count for the given string + Map eventTypeCounter = eventFingerprint.getEventTypeCounter(); + eventTypeCounter.put(str, eventTypeCounter.getOrDefault(str, 0) + 1); + } + + public int getEventCount(String str) { + // Get the count for the given string + return eventFingerprint.getEventTypeCounter().getOrDefault(str, 0); + } + + @Override + public void handleEvent(Event event) { + + addEventType(event.getEventType()); + eventFingerprint.getTimeArray().add((float) event.getTime()); + String lexicographicSortedString = toLexicographicSortedString(event); + eventFingerprint.addHashCode(lexicographicSortedString); + + + } + + private String toLexicographicSortedString(Event event) { + List strings = new ArrayList(); + for (Map.Entry e : event.getAttributes().entrySet()) { + StringBuilder tmp = new StringBuilder(); + final String key = e.getKey(); + + // dont look at cerftain attributes + switch (key) { + case Event.ATTRIBUTE_X: + case Event.ATTRIBUTE_Y: + case Event.ATTRIBUTE_TIME: + continue; + } + + tmp.append(key); + tmp.append("="); + tmp.append(e.getValue()); + strings.add(tmp.toString()); + } + Collections.sort(strings); + StringBuilder eventStr = new StringBuilder(); + for (String str : strings) { + eventStr.append(" | "); + eventStr.append(str); + } + + eventStr.append(" | "); + return eventStr.toString(); + } + + + public static void writeEventFingerprintToFile(String filePath, EventFingerprint eventFingerprint) { + try (DataOutputStream dataOutputStream = new DataOutputStream(new FileOutputStream(filePath))) { + // Write header and version + dataOutputStream.writeByte(header); + dataOutputStream.writeByte(version); + + // Write time array size and elements + dataOutputStream.writeInt(eventFingerprint.getTimeArray().size()); + for (float time : eventFingerprint.getTimeArray()) { + dataOutputStream.writeFloat(time); + } + + // Write event type counter map size and elements + dataOutputStream.writeInt(eventFingerprint.getEventTypeCounter().size()); + for (Map.Entry entry : eventFingerprint.getEventTypeCounter().entrySet()) { + dataOutputStream.writeUTF(entry.getKey()); + dataOutputStream.writeInt(entry.getValue()); + } + + // Write string hash + dataOutputStream.writeInt(eventFingerprint.getStringHash()); + } catch (IOException e) { + e.printStackTrace(); + } + } + + public static EventFingerprint readEventFingerprintFromFile(String fingerprintPath) { + EventFingerprint eventFingerprint = null; + + try (DataInputStream dataInputStream = new DataInputStream(new FileInputStream(fingerprintPath))) { + // Read header and version + byte fileHeader = dataInputStream.readByte(); + byte fileVersion = dataInputStream.readByte(); + + // Read time array + int timeArraySize = dataInputStream.readInt(); + List timeArray = new ArrayList<>(); + for (int i = 0; i < timeArraySize; i++) { + timeArray.add(dataInputStream.readFloat()); + } + + // Read event type counter map + int eventTypeCounterSize = dataInputStream.readInt(); + Map eventTypeCounter = new HashMap<>(); + for (int i = 0; i < eventTypeCounterSize; i++) { + String eventType = dataInputStream.readUTF(); + int count = dataInputStream.readInt(); + eventTypeCounter.put(eventType, count); + } + + // Read string hash + int stringHash = dataInputStream.readInt(); + + // Create EventFingerprint object + eventFingerprint = new EventFingerprint(timeArray, eventTypeCounter, stringHash); + } catch (IOException e) { + e.printStackTrace(); + } + + return eventFingerprint; + } +} From 0353e5dd40a1172c9cbad49d0a09b01fc9252b54 Mon Sep 17 00:00:00 2001 From: zlukich Date: Thu, 22 Feb 2024 11:59:48 +0100 Subject: [PATCH 03/12] fixed compare function, added printFingerprint --- .../org/matsim/core/events/EventsUtils.java | 6 +++--- .../EventsFileFingerprintComparator.java | 9 +++------ .../FingerprintEventHandler.java | 17 +++++++++++++++++ .../utils/eventsfilecomparison/Worker.java | 2 ++ 4 files changed, 25 insertions(+), 9 deletions(-) diff --git a/matsim/src/main/java/org/matsim/core/events/EventsUtils.java b/matsim/src/main/java/org/matsim/core/events/EventsUtils.java index c5bdc3f29d6..5c60e4b142a 100644 --- a/matsim/src/main/java/org/matsim/core/events/EventsUtils.java +++ b/matsim/src/main/java/org/matsim/core/events/EventsUtils.java @@ -79,16 +79,16 @@ public static void createEventsFingerprint(String eventFile, String outputFinger } - public static EventsFileComparator.Result createAndCompareEventsFingerprint(String inputFingerprint, String eventFile) { + public static EventsFileFingerprintComparator.Result createAndCompareEventsFingerprint(String inputFingerprint, String eventFile) { // header byte, version byte // bin array time stamps // event type counter map // one hash (sha1?) - new EventsFileFingerprintComparator(); + EventsFileFingerprintComparator.Result result = EventsFileFingerprintComparator.compare(inputFingerprint,eventFile); - return EventsFileComparator.Result.FILES_ARE_EQUAL; + return result; } public static void readEvents(EventsManager events, String filename) { diff --git a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparator.java b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparator.java index 94ba3b585a8..e457f7c1814 100644 --- a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparator.java +++ b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparator.java @@ -32,25 +32,22 @@ public static Result compare(final String fingerprint, final String eventsfile) FingerprintEventHandler.EventFingerprint correctFingerprint = FingerprintEventHandler.readEventFingerprintFromFile(fingerprint); - if (fingerprintFromEvents == fingerprintFromEvents) { - return Result.FILES_ARE_EQUAL; - } if (fingerprintFromEvents == null) { return Result.DIFFERENT_TIMESTEPS; } // Compare timeArray - if (!Objects.equals(fingerprintFromEvents.timeArray, fingerprintFromEvents.timeArray)) { + if (!Objects.equals(correctFingerprint.timeArray, fingerprintFromEvents.timeArray)) { return Result.DIFFERENT_TIMESTEPS; } // Compare eventTypeCounter - if (!Objects.equals(fingerprintFromEvents.eventTypeCounter, fingerprintFromEvents.eventTypeCounter)) { + if (!Objects.equals(correctFingerprint.eventTypeCounter, fingerprintFromEvents.eventTypeCounter)) { return Result.WRONG_EVENT_COUNT; } // Compare stringHash - if (!Objects.equals(fingerprintFromEvents.stringHash, fingerprintFromEvents.stringHash)) { + if (!Objects.equals(correctFingerprint.stringHash, fingerprintFromEvents.stringHash)) { return Result.MISSING_EVENT; } diff --git a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/FingerprintEventHandler.java b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/FingerprintEventHandler.java index e65ce46bdc5..e71a319ef0c 100644 --- a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/FingerprintEventHandler.java +++ b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/FingerprintEventHandler.java @@ -58,6 +58,23 @@ public void addHashCode(String stringToAdd) { stringHash += hashToAdd; } } + + public void printFingerprint(FingerprintEventHandler.EventFingerprint fingerprint) { + System.out.println("Time Array:"); + var i = 0; + for (Float value : fingerprint.getTimeArray()) { + i++; + if(i % 100000 == 0) + System.out.println(value); + } + + System.out.println("Event Type Counter:"); + for (Map.Entry entry : fingerprint.getEventTypeCounter().entrySet()) { + System.out.println(entry.getKey() + ": " + entry.getValue()); + } + + System.out.println("String Hash: " + fingerprint.getStringHash()); + } } public void addEventType(String str) { diff --git a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/Worker.java b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/Worker.java index 004a0a5e347..81c9cddf32a 100644 --- a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/Worker.java +++ b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/Worker.java @@ -144,6 +144,7 @@ private String toLexicographicSortedString(Event event) { switch( key ){ case Event.ATTRIBUTE_X: case Event.ATTRIBUTE_Y: + case Event.ATTRIBUTE_TIME: continue; } } @@ -159,6 +160,7 @@ private String toLexicographicSortedString(Event event) { eventStr.append(" | "); eventStr.append(str); } + eventStr.append(" | ") ; return eventStr.toString(); } From 29d954df4632d3460ce9eb5909a9982fec6ca6bd Mon Sep 17 00:00:00 2001 From: zlukich Date: Tue, 27 Feb 2024 16:09:26 +0100 Subject: [PATCH 04/12] changed structure, added compression, changed hash logic --- .../contrib/bicycle/run/BicycleTest.java | 2 +- .../events/VehicleLeavesTrafficEventTest.java | 6 +- ...unAverageEmissionToolOfflineExampleIT.java | 18 +- .../contrib/ev/example/RunEvExampleTest.java | 10 +- ...nEvExampleWithLTHConsumptionModelTest.java | 10 +- .../usecases/chessboard/RunChessboardIT.java | 6 +- .../run/RunParkingSearchScenarioIT.java | 10 +- .../run/RunWithParkingProxyIT.java | 6 +- .../run/RoadPricingByConfigfileTest.java | 4 +- .../builder/TravelTimeFourWaysTest.java | 5 +- .../signals/integration/SignalSystemsIT.java | 9 +- .../contrib/etaxi/run/RunETaxiScenarioIT.java | 6 +- .../taxi/optimizer/TaxiOptimizerTests.java | 6 +- .../java/playground/vsp/ev/UrbanEVIT.java | 6 +- .../org/matsim/core/events/EventsUtils.java | 34 ++- .../ComparisonResult.java | 6 + .../EventFingerprint.java | 150 ++++++++++++ .../EventsFileComparator.java | 30 ++- .../EventsFileFingerprintComparator.java | 89 ++++--- .../FingerprintEventHandler.java | 223 +++++++----------- .../java/org/matsim/examples/EquilTest.java | 5 +- .../examples/OnePercentBerlin10sIT.java | 5 +- .../org/matsim/testcases/MatsimTestUtils.java | 3 +- .../EventsFileComparatorTest.java | 2 +- .../EventsFileFingerprintComparatorTest.java | 34 +++ 25 files changed, 437 insertions(+), 248 deletions(-) create mode 100644 matsim/src/main/java/org/matsim/utils/eventsfilecomparison/ComparisonResult.java create mode 100644 matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventFingerprint.java create mode 100644 matsim/src/test/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparatorTest.java diff --git a/contribs/bicycle/src/test/java/org/matsim/contrib/bicycle/run/BicycleTest.java b/contribs/bicycle/src/test/java/org/matsim/contrib/bicycle/run/BicycleTest.java index 961e43f0182..222834c746b 100644 --- a/contribs/bicycle/src/test/java/org/matsim/contrib/bicycle/run/BicycleTest.java +++ b/contribs/bicycle/src/test/java/org/matsim/contrib/bicycle/run/BicycleTest.java @@ -65,7 +65,7 @@ import static org.junit.jupiter.api.Assertions.assertEquals; import static org.junit.jupiter.api.Assertions.assertTrue; -import static org.matsim.utils.eventsfilecomparison.EventsFileComparator.Result.FILES_ARE_EQUAL; +import static org.matsim.utils.eventsfilecomparison.ComparisonResult.FILES_ARE_EQUAL; /** * @author dziemke diff --git a/contribs/emissions/src/test/java/org/matsim/contrib/emissions/events/VehicleLeavesTrafficEventTest.java b/contribs/emissions/src/test/java/org/matsim/contrib/emissions/events/VehicleLeavesTrafficEventTest.java index 2858e572d58..1eacd47330d 100644 --- a/contribs/emissions/src/test/java/org/matsim/contrib/emissions/events/VehicleLeavesTrafficEventTest.java +++ b/contribs/emissions/src/test/java/org/matsim/contrib/emissions/events/VehicleLeavesTrafficEventTest.java @@ -19,7 +19,7 @@ import org.matsim.core.utils.io.IOUtils; import org.matsim.examples.ExamplesUtils; import org.matsim.testcases.MatsimTestUtils; -import org.matsim.utils.eventsfilecomparison.EventsFileComparator; +import org.matsim.utils.eventsfilecomparison.ComparisonResult; import java.net.URL; @@ -81,8 +81,8 @@ public void install(){ throw new RuntimeException(e) ; } final String expected = utils.getClassInputDirectory() + emissionEventsFileName; - EventsFileComparator.Result result = EventsUtils.compareEventsFiles(expected, resultingEvents); - Assertions.assertEquals( EventsFileComparator.Result.FILES_ARE_EQUAL, result); + ComparisonResult result = EventsUtils.compareEventsFiles(expected, resultingEvents); + Assertions.assertEquals( ComparisonResult.FILES_ARE_EQUAL, result); } } diff --git a/contribs/emissions/src/test/java/org/matsim/contrib/emissions/example/RunAverageEmissionToolOfflineExampleIT.java b/contribs/emissions/src/test/java/org/matsim/contrib/emissions/example/RunAverageEmissionToolOfflineExampleIT.java index 951ed2fb237..259b8472549 100644 --- a/contribs/emissions/src/test/java/org/matsim/contrib/emissions/example/RunAverageEmissionToolOfflineExampleIT.java +++ b/contribs/emissions/src/test/java/org/matsim/contrib/emissions/example/RunAverageEmissionToolOfflineExampleIT.java @@ -30,7 +30,7 @@ import org.matsim.core.utils.io.IOUtils; import org.matsim.examples.ExamplesUtils; import org.matsim.testcases.MatsimTestUtils; -import org.matsim.utils.eventsfilecomparison.EventsFileComparator.Result; +import org.matsim.utils.eventsfilecomparison.ComparisonResult; import java.net.URL; @@ -58,8 +58,8 @@ final void testAverage_vehTypeV1() { String expected = utils.getInputDirectory() + RunAverageEmissionToolOfflineExample.emissionEventsFilename; String actual = utils.getOutputDirectory() + RunAverageEmissionToolOfflineExample.emissionEventsFilename; - Result result = EventsUtils.compareEventsFiles( expected, actual ); - Assertions.assertEquals( Result.FILES_ARE_EQUAL, result); + ComparisonResult result = EventsUtils.compareEventsFiles( expected, actual ); + Assertions.assertEquals( ComparisonResult.FILES_ARE_EQUAL, result); } @Test @@ -80,8 +80,8 @@ final void testAverage_vehTypeV2() { String expected = utils.getInputDirectory() + RunAverageEmissionToolOfflineExample.emissionEventsFilename; String actual = utils.getOutputDirectory() + RunAverageEmissionToolOfflineExample.emissionEventsFilename; - Result result = EventsUtils.compareEventsFiles( expected, actual ); - Assertions.assertEquals( Result.FILES_ARE_EQUAL, result); + ComparisonResult result = EventsUtils.compareEventsFiles( expected, actual ); + Assertions.assertEquals( ComparisonResult.FILES_ARE_EQUAL, result); } /** @@ -106,8 +106,8 @@ final void testAverage_vehTypeV2b() { String expected = utils.getInputDirectory() + RunAverageEmissionToolOfflineExample.emissionEventsFilename; String actual = utils.getOutputDirectory() + RunAverageEmissionToolOfflineExample.emissionEventsFilename; - Result result = EventsUtils.compareEventsFiles( expected, actual ); - Assertions.assertEquals( Result.FILES_ARE_EQUAL, result); + ComparisonResult result = EventsUtils.compareEventsFiles( expected, actual ); + Assertions.assertEquals( ComparisonResult.FILES_ARE_EQUAL, result); } @@ -130,7 +130,7 @@ final void testAverage_vehTypeV2_HBEFA4() { String expected = utils.getInputDirectory() + RunAverageEmissionToolOfflineExample.emissionEventsFilename; String actual = utils.getOutputDirectory() + RunAverageEmissionToolOfflineExample.emissionEventsFilename; - Result result = EventsUtils.compareEventsFiles( expected, actual ); - Assertions.assertEquals( Result.FILES_ARE_EQUAL, result); + ComparisonResult result = EventsUtils.compareEventsFiles( expected, actual ); + Assertions.assertEquals( ComparisonResult.FILES_ARE_EQUAL, result); } } diff --git a/contribs/ev/src/test/java/org/matsim/contrib/ev/example/RunEvExampleTest.java b/contribs/ev/src/test/java/org/matsim/contrib/ev/example/RunEvExampleTest.java index e8bb29fea1c..87644405c7d 100644 --- a/contribs/ev/src/test/java/org/matsim/contrib/ev/example/RunEvExampleTest.java +++ b/contribs/ev/src/test/java/org/matsim/contrib/ev/example/RunEvExampleTest.java @@ -10,7 +10,7 @@ import org.matsim.core.events.EventsUtils; import org.matsim.core.population.PopulationUtils; import org.matsim.testcases.MatsimTestUtils; -import org.matsim.utils.eventsfilecomparison.EventsFileComparator; +import org.matsim.utils.eventsfilecomparison.ComparisonResult; public class RunEvExampleTest{ @@ -39,8 +39,8 @@ public class RunEvExampleTest{ { String expected = utils.getInputDirectory() + "/output_events.xml.gz" ; String actual = utils.getOutputDirectory() + "/output_events.xml.gz" ; - EventsFileComparator.Result result = EventsUtils.compareEventsFiles( expected, actual ); - Assertions.assertEquals(EventsFileComparator.Result.FILES_ARE_EQUAL, result); + ComparisonResult result = EventsUtils.compareEventsFiles( expected, actual ); + Assertions.assertEquals(ComparisonResult.FILES_ARE_EQUAL, result); } } catch ( Exception ee ) { @@ -72,8 +72,8 @@ public class RunEvExampleTest{ { String expected = utils.getInputDirectory() + "/output_events.xml.gz" ; String actual = utils.getOutputDirectory() + "/output_events.xml.gz" ; - EventsFileComparator.Result result = EventsUtils.compareEventsFiles( expected, actual ); - Assertions.assertEquals(EventsFileComparator.Result.FILES_ARE_EQUAL, result); + ComparisonResult result = EventsUtils.compareEventsFiles( expected, actual ); + Assertions.assertEquals(ComparisonResult.FILES_ARE_EQUAL, result); } } catch ( Exception ee ) { diff --git a/contribs/ev/src/test/java/org/matsim/contrib/ev/example/RunEvExampleWithLTHConsumptionModelTest.java b/contribs/ev/src/test/java/org/matsim/contrib/ev/example/RunEvExampleWithLTHConsumptionModelTest.java index ffc5f81e296..447b79966ad 100644 --- a/contribs/ev/src/test/java/org/matsim/contrib/ev/example/RunEvExampleWithLTHConsumptionModelTest.java +++ b/contribs/ev/src/test/java/org/matsim/contrib/ev/example/RunEvExampleWithLTHConsumptionModelTest.java @@ -10,11 +10,7 @@ import org.matsim.core.events.EventsUtils; import org.matsim.core.population.PopulationUtils; import org.matsim.testcases.MatsimTestUtils; -import org.matsim.utils.eventsfilecomparison.EventsFileComparator; - -import java.io.File; -import java.net.MalformedURLException; -import java.net.URL; +import org.matsim.utils.eventsfilecomparison.ComparisonResult; public class RunEvExampleWithLTHConsumptionModelTest{ @@ -43,8 +39,8 @@ void runTest(){ { String expected = utils.getInputDirectory() + "/output_events.xml.gz" ; String actual = utils.getOutputDirectory() + "/output_events.xml.gz" ; - EventsFileComparator.Result result = EventsUtils.compareEventsFiles( expected, actual ); - Assertions.assertEquals( EventsFileComparator.Result.FILES_ARE_EQUAL, result ); + ComparisonResult result = EventsUtils.compareEventsFiles( expected, actual ); + Assertions.assertEquals( ComparisonResult.FILES_ARE_EQUAL, result ); } } catch ( Exception ee ) { diff --git a/contribs/freight/src/test/java/org/matsim/freight/carriers/usecases/chessboard/RunChessboardIT.java b/contribs/freight/src/test/java/org/matsim/freight/carriers/usecases/chessboard/RunChessboardIT.java index ced181d3172..b195b3b89d7 100644 --- a/contribs/freight/src/test/java/org/matsim/freight/carriers/usecases/chessboard/RunChessboardIT.java +++ b/contribs/freight/src/test/java/org/matsim/freight/carriers/usecases/chessboard/RunChessboardIT.java @@ -32,7 +32,7 @@ import org.matsim.core.utils.io.IOUtils; import org.matsim.examples.ExamplesUtils; import org.matsim.testcases.MatsimTestUtils; -import org.matsim.utils.eventsfilecomparison.EventsFileComparator; +import org.matsim.utils.eventsfilecomparison.ComparisonResult; public class RunChessboardIT { @@ -60,8 +60,8 @@ void runChessboard() { { String expected = utils.getInputDirectory() + "/output_events.xml.gz" ; String actual = utils.getOutputDirectory() + "/output_events.xml.gz" ; - EventsFileComparator.Result result = EventsUtils.compareEventsFiles( expected, actual ); - Assertions.assertEquals( EventsFileComparator.Result.FILES_ARE_EQUAL, result ); + ComparisonResult result = EventsUtils.compareEventsFiles( expected, actual ); + Assertions.assertEquals( ComparisonResult.FILES_ARE_EQUAL, result ); } } catch (Exception ee ) { ee.printStackTrace(); diff --git a/contribs/parking/src/test/java/org/matsim/contrib/parking/parkingchoice/run/RunParkingSearchScenarioIT.java b/contribs/parking/src/test/java/org/matsim/contrib/parking/parkingchoice/run/RunParkingSearchScenarioIT.java index fa0dc4f5ffd..e3959961195 100644 --- a/contribs/parking/src/test/java/org/matsim/contrib/parking/parkingchoice/run/RunParkingSearchScenarioIT.java +++ b/contribs/parking/src/test/java/org/matsim/contrib/parking/parkingchoice/run/RunParkingSearchScenarioIT.java @@ -33,7 +33,7 @@ import org.matsim.core.events.EventsUtils; import org.matsim.core.population.PopulationUtils; import org.matsim.testcases.MatsimTestUtils; -import org.matsim.utils.eventsfilecomparison.EventsFileComparator; +import org.matsim.utils.eventsfilecomparison.ComparisonResult; /** * @author jbischoff @@ -108,8 +108,8 @@ void testRunParkingDistanceMemoryStrategy() { { String expected = utils.getInputDirectory() + "/output_events.xml.gz"; String actual = utils.getOutputDirectory() + "/output_events.xml.gz"; - EventsFileComparator.Result result = EventsUtils.compareEventsFiles(expected, actual); - Assertions.assertEquals(EventsFileComparator.Result.FILES_ARE_EQUAL, result); + ComparisonResult result = EventsUtils.compareEventsFiles(expected, actual); + Assertions.assertEquals(ComparisonResult.FILES_ARE_EQUAL, result); } } catch (Exception e) { e.printStackTrace(); @@ -146,8 +146,8 @@ void testRunParkingNearestParkingSpotStrategy() { { String expected = utils.getInputDirectory() + "/output_events.xml.gz"; String actual = utils.getOutputDirectory() + "/output_events.xml.gz"; - EventsFileComparator.Result result = EventsUtils.compareEventsFiles(expected, actual); - Assertions.assertEquals(EventsFileComparator.Result.FILES_ARE_EQUAL, result); + ComparisonResult result = EventsUtils.compareEventsFiles(expected, actual); + Assertions.assertEquals(ComparisonResult.FILES_ARE_EQUAL, result); } } catch (Exception e) { e.printStackTrace(); diff --git a/contribs/parking/src/test/java/org/matsim/contrib/parking/parkingproxy/run/RunWithParkingProxyIT.java b/contribs/parking/src/test/java/org/matsim/contrib/parking/parkingproxy/run/RunWithParkingProxyIT.java index 6d17839c44c..e87a4afbaba 100644 --- a/contribs/parking/src/test/java/org/matsim/contrib/parking/parkingproxy/run/RunWithParkingProxyIT.java +++ b/contribs/parking/src/test/java/org/matsim/contrib/parking/parkingproxy/run/RunWithParkingProxyIT.java @@ -30,7 +30,7 @@ import org.matsim.core.utils.io.IOUtils; import org.matsim.examples.ExamplesUtils; import org.matsim.testcases.MatsimTestUtils; -import org.matsim.utils.eventsfilecomparison.EventsFileComparator.Result; +import org.matsim.utils.eventsfilecomparison.ComparisonResult; public class RunWithParkingProxyIT { private static final Logger log = LogManager.getLogger(RunWithParkingProxyIT.class); @@ -50,8 +50,8 @@ void testMain(){ { String expected = utils.getInputDirectory() + "/output_events.xml.gz" ; String actual = utils.getOutputDirectory() + "/output_events.xml.gz" ; - Result result = EventsUtils.compareEventsFiles( expected, actual ); - if(!result.equals(Result.FILES_ARE_EQUAL)) { + ComparisonResult result = EventsUtils.compareEventsFiles( expected, actual ); + if(!result.equals(ComparisonResult.FILES_ARE_EQUAL)) { throw new RuntimeException("Events comparison ended with result " + result.name()); } } diff --git a/contribs/roadpricing/src/test/java/org/matsim/contrib/roadpricing/run/RoadPricingByConfigfileTest.java b/contribs/roadpricing/src/test/java/org/matsim/contrib/roadpricing/run/RoadPricingByConfigfileTest.java index 70b4bc9344d..e59e74dd90b 100644 --- a/contribs/roadpricing/src/test/java/org/matsim/contrib/roadpricing/run/RoadPricingByConfigfileTest.java +++ b/contribs/roadpricing/src/test/java/org/matsim/contrib/roadpricing/run/RoadPricingByConfigfileTest.java @@ -32,7 +32,7 @@ import org.matsim.core.utils.io.IOUtils; import org.matsim.examples.ExamplesUtils; import org.matsim.testcases.MatsimTestUtils; -import org.matsim.utils.eventsfilecomparison.EventsFileComparator; +import org.matsim.utils.eventsfilecomparison.ComparisonResult; /** * @author vsp-gleich @@ -55,7 +55,7 @@ final void testMain() { { String expected = utils.getInputDirectory() + "/output_events.xml.gz" ; String actual = utils.getOutputDirectory() + "/output_events.xml.gz" ; - Assertions.assertEquals(EventsFileComparator.Result.FILES_ARE_EQUAL, EventsUtils.compareEventsFiles( expected, actual )); + Assertions.assertEquals(ComparisonResult.FILES_ARE_EQUAL, EventsUtils.compareEventsFiles( expected, actual )); } { final Population expected = PopulationUtils.createPopulation( ConfigUtils.createConfig() ); diff --git a/contribs/signals/src/test/java/org/matsim/contrib/signals/builder/TravelTimeFourWaysTest.java b/contribs/signals/src/test/java/org/matsim/contrib/signals/builder/TravelTimeFourWaysTest.java index 9e341d444a6..ba97489f89e 100644 --- a/contribs/signals/src/test/java/org/matsim/contrib/signals/builder/TravelTimeFourWaysTest.java +++ b/contribs/signals/src/test/java/org/matsim/contrib/signals/builder/TravelTimeFourWaysTest.java @@ -38,8 +38,7 @@ import org.matsim.core.scenario.ScenarioUtils; import org.matsim.testcases.MatsimTestUtils; import org.matsim.utils.eventsfilecomparison.EventsFileComparator; - -import static org.matsim.utils.eventsfilecomparison.EventsFileComparator.*; +import org.matsim.utils.eventsfilecomparison.ComparisonResult; /** * @author aneumann @@ -125,7 +124,7 @@ private void runQSimWithSignals(final Scenario scenario) { eventsXmlWriter.closeFile(); // Assert.assertEquals("different events files", EventsFileComparator.compareAndReturnInt(this.testUtils.getInputDirectory() + EVENTSFILE, eventsOut), 0); - Assertions.assertEquals( Result.FILES_ARE_EQUAL, new EventsFileComparator().setIgnoringCoordinates( true ).runComparison( this.testUtils.getInputDirectory() + EVENTSFILE, eventsOut ) ); + Assertions.assertEquals( ComparisonResult.FILES_ARE_EQUAL, new EventsFileComparator().setIgnoringCoordinates( true ).runComparison( this.testUtils.getInputDirectory() + EVENTSFILE, eventsOut ) ); } } diff --git a/contribs/signals/src/test/java/org/matsim/contrib/signals/integration/SignalSystemsIT.java b/contribs/signals/src/test/java/org/matsim/contrib/signals/integration/SignalSystemsIT.java index 49e8af1f140..b44a5cf35ea 100644 --- a/contribs/signals/src/test/java/org/matsim/contrib/signals/integration/SignalSystemsIT.java +++ b/contribs/signals/src/test/java/org/matsim/contrib/signals/integration/SignalSystemsIT.java @@ -36,6 +36,7 @@ import org.matsim.core.scenario.ScenarioUtils; import org.matsim.testcases.MatsimTestUtils; import org.matsim.utils.eventsfilecomparison.EventsFileComparator; +import org.matsim.utils.eventsfilecomparison.ComparisonResult; import java.io.File; @@ -88,7 +89,7 @@ void testSignalSystems() { //iteration 0 String iterationOutput = controlerOutputDir + "ITERS/it.0/"; - Assertions.assertEquals(EventsFileComparator.Result.FILES_ARE_EQUAL, + Assertions.assertEquals(ComparisonResult.FILES_ARE_EQUAL, new EventsFileComparator().setIgnoringCoordinates( true ).runComparison( inputDirectory + "0.events.xml.gz", iterationOutput + "0.events.xml.gz"), "different events files after iteration 0 " @@ -113,7 +114,7 @@ void testSignalSystems() { //iteration 10 String iterationOutput = controlerOutputDir + "ITERS/it.10/"; - Assertions.assertEquals(EventsFileComparator.Result.FILES_ARE_EQUAL, + Assertions.assertEquals(ComparisonResult.FILES_ARE_EQUAL, new EventsFileComparator().setIgnoringCoordinates( true ).runComparison( inputDirectory + "10.events.xml.gz", iterationOutput + "10.events.xml.gz" ), "different event files after iteration 10" ); @@ -181,7 +182,7 @@ void testSignalSystemsWTryEndTimeThenDuration() { //iteration 0 String iterationOutput = controlerOutputDir + "ITERS/it.0/"; - Assertions.assertEquals(EventsFileComparator.Result.FILES_ARE_EQUAL, + Assertions.assertEquals(ComparisonResult.FILES_ARE_EQUAL, new EventsFileComparator().setIgnoringCoordinates( true ).runComparison( inputDirectory + "0.events.xml.gz", iterationOutput + "0.events.xml.gz"), "different events files after iteration 0 " @@ -206,7 +207,7 @@ void testSignalSystemsWTryEndTimeThenDuration() { //iteration 10 String iterationOutput = controlerOutputDir + "ITERS/it.10/"; - Assertions.assertEquals(EventsFileComparator.Result.FILES_ARE_EQUAL, + Assertions.assertEquals(ComparisonResult.FILES_ARE_EQUAL, new EventsFileComparator().setIgnoringCoordinates( true ).runComparison( inputDirectory + "10.events.xml.gz", iterationOutput + "10.events.xml.gz"), "different event files after iteration 10" ); diff --git a/contribs/taxi/src/test/java/org/matsim/contrib/etaxi/run/RunETaxiScenarioIT.java b/contribs/taxi/src/test/java/org/matsim/contrib/etaxi/run/RunETaxiScenarioIT.java index c086adedc50..5e029c8a29e 100644 --- a/contribs/taxi/src/test/java/org/matsim/contrib/etaxi/run/RunETaxiScenarioIT.java +++ b/contribs/taxi/src/test/java/org/matsim/contrib/etaxi/run/RunETaxiScenarioIT.java @@ -29,7 +29,7 @@ import org.matsim.core.utils.io.IOUtils; import org.matsim.examples.ExamplesUtils; import org.matsim.testcases.MatsimTestUtils; -import org.matsim.utils.eventsfilecomparison.EventsFileComparator; +import org.matsim.utils.eventsfilecomparison.ComparisonResult; /** * @author michalm @@ -73,8 +73,8 @@ private void runScenario(String configPath) { { String expected = utils.getInputDirectory() + "/output_events.xml.gz"; String actual = utils.getOutputDirectory() + "/output_events.xml.gz"; - EventsFileComparator.Result result = EventsUtils.compareEventsFiles(expected, actual); - Assertions.assertEquals(EventsFileComparator.Result.FILES_ARE_EQUAL, result); + ComparisonResult result = EventsUtils.compareEventsFiles(expected, actual); + Assertions.assertEquals(ComparisonResult.FILES_ARE_EQUAL, result); } } } diff --git a/contribs/taxi/src/test/java/org/matsim/contrib/taxi/optimizer/TaxiOptimizerTests.java b/contribs/taxi/src/test/java/org/matsim/contrib/taxi/optimizer/TaxiOptimizerTests.java index ca985585f82..3a3800467af 100644 --- a/contribs/taxi/src/test/java/org/matsim/contrib/taxi/optimizer/TaxiOptimizerTests.java +++ b/contribs/taxi/src/test/java/org/matsim/contrib/taxi/optimizer/TaxiOptimizerTests.java @@ -35,7 +35,7 @@ import org.matsim.core.utils.io.IOUtils; import org.matsim.examples.ExamplesUtils; import org.matsim.testcases.MatsimTestUtils; -import org.matsim.utils.eventsfilecomparison.EventsFileComparator; +import org.matsim.utils.eventsfilecomparison.ComparisonResult; public class TaxiOptimizerTests { public static void runBenchmark(boolean vehicleDiversion, AbstractTaxiOptimizerParams taxiOptimizerParams, MatsimTestUtils utils) { @@ -70,8 +70,8 @@ public static void runBenchmark(boolean vehicleDiversion, AbstractTaxiOptimizerP { String expected = utils.getInputDirectory() + "/output_events.xml.gz"; String actual = utils.getOutputDirectory() + "/output_events.xml.gz"; - EventsFileComparator.Result result = EventsUtils.compareEventsFiles(expected, actual); - Assertions.assertEquals(EventsFileComparator.Result.FILES_ARE_EQUAL, result); + ComparisonResult result = EventsUtils.compareEventsFiles(expected, actual); + Assertions.assertEquals(ComparisonResult.FILES_ARE_EQUAL, result); } } } diff --git a/contribs/vsp/src/test/java/playground/vsp/ev/UrbanEVIT.java b/contribs/vsp/src/test/java/playground/vsp/ev/UrbanEVIT.java index 823fb89bf83..b96c04fb079 100644 --- a/contribs/vsp/src/test/java/playground/vsp/ev/UrbanEVIT.java +++ b/contribs/vsp/src/test/java/playground/vsp/ev/UrbanEVIT.java @@ -9,7 +9,7 @@ import org.matsim.core.events.EventsUtils; import org.matsim.core.population.PopulationUtils; import org.matsim.testcases.MatsimTestUtils; -import org.matsim.utils.eventsfilecomparison.EventsFileComparator; +import org.matsim.utils.eventsfilecomparison.ComparisonResult; public class UrbanEVIT { @RegisterExtension private MatsimTestUtils utils = new MatsimTestUtils(); @@ -38,8 +38,8 @@ void run() { { String expected = utils.getInputDirectory() + "/output_events.xml.gz" ; String actual = utils.getOutputDirectory() + "/output_events.xml.gz" ; - EventsFileComparator.Result result = EventsUtils.compareEventsFiles( expected, actual ); - Assertions.assertEquals( EventsFileComparator.Result.FILES_ARE_EQUAL, result ); + ComparisonResult result = EventsUtils.compareEventsFiles( expected, actual ); + Assertions.assertEquals( ComparisonResult.FILES_ARE_EQUAL, result ); } } catch ( Exception ee ) { diff --git a/matsim/src/main/java/org/matsim/core/events/EventsUtils.java b/matsim/src/main/java/org/matsim/core/events/EventsUtils.java index 5c60e4b142a..2f02eb3c6a9 100644 --- a/matsim/src/main/java/org/matsim/core/events/EventsUtils.java +++ b/matsim/src/main/java/org/matsim/core/events/EventsUtils.java @@ -21,12 +21,11 @@ package org.matsim.core.events; +import org.apache.commons.compress.utils.FileNameUtils; import org.matsim.core.api.experimental.events.EventsManager; import org.matsim.core.config.Config; import org.matsim.core.controler.Injector; -import org.matsim.utils.eventsfilecomparison.EventsFileComparator; -import org.matsim.utils.eventsfilecomparison.EventsFileFingerprintComparator; -import org.matsim.utils.eventsfilecomparison.FingerprintEventHandler; +import org.matsim.utils.eventsfilecomparison.*; public final class EventsUtils { @@ -67,37 +66,46 @@ public static EventsManager getParallelFeedableInstance(EventsManager events) { } } + /** + * Create and write fingerprint file for events. + */ public static void createEventsFingerprint(String eventFile, String outputFingerprintFile) { - EventsManager manager = createEventsManager(); + EventsManager manager = createEventsManager(); FingerprintEventHandler fingerprintEventHandler = new FingerprintEventHandler(); manager.addHandler(fingerprintEventHandler); - EventsUtils.readEvents(manager, eventFile); + manager.finishProcessing(); - FingerprintEventHandler.writeEventFingerprintToFile(outputFingerprintFile,fingerprintEventHandler.eventFingerprint); - + EventFingerprint.write(outputFingerprintFile, fingerprintEventHandler.eventFingerprint); } - public static EventsFileFingerprintComparator.Result createAndCompareEventsFingerprint(String inputFingerprint, String eventFile) { + + /** + * Compares existing event file against fingerprint file. This will also create new fingerprint file along the input events. + * + * @return comparison results + */ + public static ComparisonResult createAndCompareEventsFingerprint(String inputFingerprint, String eventFile) { // header byte, version byte // bin array time stamps // event type counter map // one hash (sha1?) - EventsFileFingerprintComparator.Result result = EventsFileFingerprintComparator.compare(inputFingerprint,eventFile); + String baseName = FileNameUtils.getBaseName(eventFile).replace(".xml", ""); + + createEventsFingerprint(eventFile, baseName + ".fp.zst"); - return result; + return EventsFileFingerprintComparator.compare(inputFingerprint, eventFile); } public static void readEvents(EventsManager events, String filename) { new MatsimEventsReader(events).readFile(filename); } - public static EventsFileComparator.Result compareEventsFiles(String filename1, String filename2) { - EventsFileComparator.Result result = EventsFileComparator.compare(filename1, filename2); - return result; + public static ComparisonResult compareEventsFiles(String filename1, String filename2) { + return EventsFileComparator.compare(filename1, filename2); } } diff --git a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/ComparisonResult.java b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/ComparisonResult.java new file mode 100644 index 00000000000..c77394fddd4 --- /dev/null +++ b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/ComparisonResult.java @@ -0,0 +1,6 @@ +package org.matsim.utils.eventsfilecomparison; + +/** + * Result of event file comparison. + */ +public enum ComparisonResult {FILES_ARE_EQUAL, DIFFERENT_NUMBER_OF_TIMESTEPS, DIFFERENT_TIMESTEPS, DIFFERENT_EVENT_ATTRIBUTES, MISSING_EVENT, WRONG_EVENT_COUNT} diff --git a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventFingerprint.java b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventFingerprint.java new file mode 100644 index 00000000000..71ca4b5697d --- /dev/null +++ b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventFingerprint.java @@ -0,0 +1,150 @@ +package org.matsim.utils.eventsfilecomparison; + +import it.unimi.dsi.fastutil.floats.FloatArrayList; +import it.unimi.dsi.fastutil.floats.FloatList; +import it.unimi.dsi.fastutil.objects.Object2IntMap; +import it.unimi.dsi.fastutil.objects.Object2IntOpenHashMap; +import org.matsim.core.utils.io.IOUtils; + +import java.io.*; +import java.nio.charset.StandardCharsets; +import java.security.MessageDigest; +import java.security.NoSuchAlgorithmException; +import java.util.*; + +public final class EventFingerprint { + + /** + * Header for version 1, FP/1 + */ + static final int HEADER_V1 = 0x46502f31; + + final FloatList timeArray; + final Object2IntMap eventTypeCounter; + final byte[] hash; + + /** + * Builder for the hash. + */ + private final MessageDigest digest; + + private EventFingerprint(FloatList timeArray, Object2IntMap eventTypeCounter, byte[] hash) { + this.timeArray = timeArray; + this.eventTypeCounter = eventTypeCounter; + this.hash = hash; + this.digest = null; + } + + public EventFingerprint() { + this.timeArray = new FloatArrayList(); + this.eventTypeCounter = new Object2IntOpenHashMap<>(); + this.hash = new byte[20]; + + try { + digest = MessageDigest.getInstance("SHA-1"); + } catch (NoSuchAlgorithmException e) { + throw new IllegalStateException("Hashing not supported;"); + } + } + + public void addTimeStamp(double timestamp) { + timeArray.add((float) timestamp); + } + + public void addEventType(String str) { + // Increment the count for the given string + eventTypeCounter.mergeInt(str, 1, Integer::sum); + } + + public void addHashCode(String stringToAdd) { + if (stringToAdd == null) { + return; + } + + digest.update(stringToAdd.getBytes(StandardCharsets.UTF_8)); + } + + byte[] computeHash() { + if (this.digest == null) + throw new IllegalStateException("Hash was from from input and can not be computed"); + + byte[] digest = this.digest.digest(); + System.arraycopy(digest, 0, hash, 0, hash.length); + return hash; + } + + public static void printFingerprint(EventFingerprint fingerprint) { + System.out.println("Time Array:"); + var i = 0; + for (Float value : fingerprint.timeArray) { + System.out.print(value); + } + + System.out.println("Event Type Counter:"); + for (Map.Entry entry : fingerprint.eventTypeCounter.entrySet()) { + System.out.println(entry.getKey() + ": " + entry.getValue()); + } + + System.out.println("String Hash: " + Arrays.toString(fingerprint.hash)); + } + + public static void write(String filePath, EventFingerprint eventFingerprint) { + try (DataOutputStream dataOutputStream = new DataOutputStream(IOUtils.getOutputStream(IOUtils.getFileUrl(filePath), false))) { + // Write header and version + dataOutputStream.writeInt(EventFingerprint.HEADER_V1); + + // Write time array size and elements + dataOutputStream.writeInt(eventFingerprint.timeArray.size()); + for (float time : eventFingerprint.timeArray) { + dataOutputStream.writeFloat(time); + } + + // Write event type counter map size and elements + dataOutputStream.writeInt(eventFingerprint.eventTypeCounter.size()); + for (Map.Entry entry : eventFingerprint.eventTypeCounter.entrySet()) { + dataOutputStream.writeUTF(entry.getKey()); + dataOutputStream.writeInt(entry.getValue()); + } + + // Write byte hash + dataOutputStream.write(eventFingerprint.computeHash()); + } catch (IOException e) { + e.printStackTrace(); + } + } + + public static EventFingerprint read(String fingerprintPath) { + EventFingerprint eventFingerprint = null; + + try (DataInputStream dataInputStream = new DataInputStream(IOUtils.getInputStream(IOUtils.getFileUrl(fingerprintPath)))) { + // Read header and version + int fileHeader = dataInputStream.readInt(); + + // Read time array + int timeArraySize = dataInputStream.readInt(); + FloatList timeArray = new FloatArrayList(); + for (int i = 0; i < timeArraySize; i++) { + timeArray.add(dataInputStream.readFloat()); + } + + // Read event type counter map + int eventTypeCounterSize = dataInputStream.readInt(); + Object2IntMap eventTypeCounter = new Object2IntOpenHashMap<>(); + for (int i = 0; i < eventTypeCounterSize; i++) { + String eventType = dataInputStream.readUTF(); + int count = dataInputStream.readInt(); + eventTypeCounter.put(eventType, count); + } + + // Read string hash + byte[] hash = dataInputStream.readNBytes(20); + + // Create EventFingerprint object + eventFingerprint = new EventFingerprint(timeArray, eventTypeCounter, hash); + } catch (IOException e) { + throw new UncheckedIOException(e); + } + + return eventFingerprint; + } +} diff --git a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileComparator.java b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileComparator.java index 66c44705ca7..90c8d6e4e0c 100644 --- a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileComparator.java +++ b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileComparator.java @@ -37,8 +37,6 @@ public final class EventsFileComparator { private static final Logger log = LogManager.getLogger(EventsFileComparator.class); - public enum Result { FILES_ARE_EQUAL, DIFFERENT_NUMBER_OF_TIMESTEPS, DIFFERENT_TIMESTEPS, MISSING_EVENT, WRONG_EVENT_COUNT } - private boolean ignoringCoordinates = false; public EventsFileComparator setIgnoringCoordinates( boolean ignoringCoordinates ){ this.ignoringCoordinates = ignoringCoordinates; @@ -62,13 +60,13 @@ public static void main(String[] args) { * * @param filename1 name of the first event file * @param filename2 name of the second event file - * @return Result.FILES_ARE_EQUAL if the events files are equal, or some error code (see {@link Result}) if not. + * @return Result.FILES_ARE_EQUAL if the events files are equal, or some error code (see {@link ComparisonResult}) if not. */ - public static Result compare(final String filename1, final String filename2) { + public static ComparisonResult compare(final String filename1, final String filename2) { return new EventsFileComparator().runComparison( filename1, filename2 ); } - public Result runComparison( final String filename1, final String filename2 ) { + public ComparisonResult runComparison(final String filename1, final String filename2 ) { // (need method name different from pre-existing static method. kai, feb'20) EventsComparator comparator = new EventsComparator( ); @@ -90,8 +88,8 @@ public Result runComparison( final String filename1, final String filename2 ) { e.printStackTrace(); } - Result retCode = comparator.retCode; - if (retCode == Result.FILES_ARE_EQUAL) { + ComparisonResult retCode = comparator.retCode; + if (retCode == ComparisonResult.FILES_ARE_EQUAL) { log.info("Event files are semantically equivalent."); } else { log.warn("Event files differ."); @@ -103,7 +101,7 @@ private static class EventsComparator implements Runnable { private Worker worker1 = null; private Worker worker2 = null; - private volatile Result retCode = null ; + private volatile ComparisonResult retCode = null ; /*package*/ void setWorkers(final Worker w1, final Worker w2) { this.worker1 = w1; @@ -114,13 +112,13 @@ private static class EventsComparator implements Runnable { public void run() { if (this.worker1.getCurrentTime() != this.worker2.getCurrentTime()) { log.warn("Differnt time steps in event files!"); - setExitCode(Result.DIFFERENT_TIMESTEPS); + setExitCode(ComparisonResult.DIFFERENT_TIMESTEPS); return; } if (this.worker1.isFinished() != this.worker2.isFinished()) { log.warn("Events files have different number of time steps!"); - setExitCode(Result.DIFFERENT_NUMBER_OF_TIMESTEPS); + setExitCode(ComparisonResult.DIFFERENT_NUMBER_OF_TIMESTEPS); return; } @@ -141,7 +139,7 @@ public void run() { log.warn("The event:"); log.warn(entry.getKey()); log.warn("is missing in events file:" + worker2.getEventsFile()); - setExitCode(Result.MISSING_EVENT); + setExitCode(ComparisonResult.MISSING_EVENT); problem = true; if (logCounter == 50) { log.warn(Gbl.FUTURE_SUPPRESSED); @@ -152,7 +150,7 @@ public void run() { log.warn( "Wrong event count for: " + entry.getKey() + "\n" + entry.getValue().getCount() + " times in file:" + worker1.getEventsFile() + "\n" + counter.getCount() + " times in file:" + worker2.getEventsFile() ); - setExitCode( Result.WRONG_EVENT_COUNT ); + setExitCode( ComparisonResult.WRONG_EVENT_COUNT ); problem = true; } } @@ -168,7 +166,7 @@ public void run() { log.warn("The event:"); log.warn(e.getKey()); log.warn("is missing in events file:" + worker1.getEventsFile()); - setExitCode(Result.MISSING_EVENT); + setExitCode(ComparisonResult.MISSING_EVENT); problem = true; if (logCounter == 50) { log.warn(Gbl.FUTURE_SUPPRESSED); @@ -182,13 +180,13 @@ public void run() { } if (this.worker1.isFinished()) { - setExitCode(Result.FILES_ARE_EQUAL); + setExitCode(ComparisonResult.FILES_ARE_EQUAL); } } - private void setExitCode(final Result errCode) { + private void setExitCode(final ComparisonResult errCode) { this.retCode= errCode; - if (errCode != Result.FILES_ARE_EQUAL) { + if (errCode != ComparisonResult.FILES_ARE_EQUAL) { this.worker1.interrupt(); this.worker2.interrupt(); } diff --git a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparator.java b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparator.java index e457f7c1814..d6984cca371 100644 --- a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparator.java +++ b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparator.java @@ -1,60 +1,95 @@ package org.matsim.utils.eventsfilecomparison; -import it.unimi.dsi.fastutil.Hash; +import it.unimi.dsi.fastutil.objects.Object2IntMap; +import it.unimi.dsi.fastutil.objects.ObjectIterator; import org.apache.logging.log4j.LogManager; import org.apache.logging.log4j.Logger; -import org.matsim.api.core.v01.events.Event; import org.matsim.core.api.experimental.events.EventsManager; -import org.matsim.core.events.EventsManagerImpl; import org.matsim.core.events.EventsUtils; -import org.matsim.core.events.handler.BasicEventHandler; -import java.io.*; import java.util.*; -import java.util.concurrent.CyclicBarrier; public class EventsFileFingerprintComparator { private static final Logger log = LogManager.getLogger(EventsFileComparator.class); - public enum Result { FILES_ARE_EQUAL, DIFFERENT_NUMBER_OF_TIMESTEPS, DIFFERENT_TIMESTEPS, MISSING_EVENT, WRONG_EVENT_COUNT } - public static Result compare(final String fingerprint, final String eventsfile) { - FingerprintEventHandler handler = new FingerprintEventHandler(); + public static ComparisonResult compare(final String fingerprint, final String eventsfile) { + + EventFingerprint correctFingerprint = EventFingerprint.read(fingerprint); + + FingerprintEventHandler handler = new FingerprintEventHandler(correctFingerprint); EventsManager manager = EventsUtils.createEventsManager(); manager.addHandler(handler); - EventsUtils.readEvents(manager,eventsfile); + EventsUtils.readEvents(manager, eventsfile); + + manager.finishProcessing(); - FingerprintEventHandler.EventFingerprint fingerprintFromEvents = handler.eventFingerprint; + ComparisonResult result = handler.finishProcessing(); - FingerprintEventHandler.EventFingerprint correctFingerprint = FingerprintEventHandler.readEventFingerprintFromFile(fingerprint); + log.warn(handler.comparisonMessage); + // All fields are equal + return result; + } - if (fingerprintFromEvents == null) { - return Result.DIFFERENT_TIMESTEPS; + public static ComparisonResult compareFingerprints(final String fp1, final String fp2) { + + + EventFingerprint fingerprint1 = EventFingerprint.read(fp1); + EventFingerprint fingerprint2 = EventFingerprint.read(fp2); + + String log_message = ""; + //Check if time array size is the same + if(fingerprint1.timeArray.size()!=fingerprint2.timeArray.size()){ + log_message = "Different number of timesteps"; + log.warn(log_message); + return ComparisonResult.DIFFERENT_NUMBER_OF_TIMESTEPS; } - // Compare timeArray - if (!Objects.equals(correctFingerprint.timeArray, fingerprintFromEvents.timeArray)) { - return Result.DIFFERENT_TIMESTEPS; + //Check if both time arrays have the same timesteps + if(!Arrays.equals(fingerprint1.timeArray.toFloatArray(),fingerprint2.timeArray.toFloatArray())){ + log_message = "Different timesteps"; + log.warn(log_message); + return ComparisonResult.DIFFERENT_TIMESTEPS; } - // Compare eventTypeCounter - if (!Objects.equals(correctFingerprint.eventTypeCounter, fingerprintFromEvents.eventTypeCounter)) { - return Result.WRONG_EVENT_COUNT; + + //Check which event type counts are different among 2 fingerprints + boolean countDiffers = false; + for (Object2IntMap.Entry entry1 : fingerprint1.eventTypeCounter.object2IntEntrySet()) { + String key = entry1.getKey(); + int count1 = entry1.getIntValue(); + int count2 = fingerprint2.eventTypeCounter.getInt(key); + if (count1 != count2) { + countDiffers = true; + log_message = log_message + ("\r\nCount for key '" + key + "' differs: " + count1 + " != " + count2); + } } + if(countDiffers){ + log.warn(log_message); + return ComparisonResult.WRONG_EVENT_COUNT; + } + + + //Check if total hash is the same + byte[] hash1 = fingerprint1.hash; + byte[] hash2 = fingerprint2.hash; + if(!Arrays.equals(hash1, hash2)){ + + log_message = String.format("Difference occurred hash codes hash of first file is %s, hash of second is %s", Arrays.toString(hash1), Arrays.toString(hash2)); - // Compare stringHash - if (!Objects.equals(correctFingerprint.stringHash, fingerprintFromEvents.stringHash)) { - return Result.MISSING_EVENT; + log.warn(log_message); + return ComparisonResult.MISSING_EVENT; } - // All fields are equal - return Result.FILES_ARE_EQUAL; - } + log_message = "Files are equal"; + log.warn(log_message); + // TODO: compare and generate messages + return ComparisonResult.FILES_ARE_EQUAL; + } } - diff --git a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/FingerprintEventHandler.java b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/FingerprintEventHandler.java index e71a319ef0c..030d4b9bb82 100644 --- a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/FingerprintEventHandler.java +++ b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/FingerprintEventHandler.java @@ -1,101 +1,123 @@ package org.matsim.utils.eventsfilecomparison; -import it.unimi.dsi.fastutil.Hash; +import it.unimi.dsi.fastutil.floats.FloatListIterator; +import it.unimi.dsi.fastutil.objects.Object2IntMap; +import it.unimi.dsi.fastutil.objects.ObjectIterator; import org.matsim.api.core.v01.events.Event; import org.matsim.core.events.handler.BasicEventHandler; +import org.matsim.counts.algorithms.graphs.helper.Comp; -import java.io.*; -import java.security.MessageDigest; import java.util.*; -public class FingerprintEventHandler implements BasicEventHandler { +public final class FingerprintEventHandler implements BasicEventHandler { - public boolean ignoringCoordinates = false; - public static byte header = 1; - public static byte version = 1; + /** + * Generated finger print. + */ public final EventFingerprint eventFingerprint = new EventFingerprint(); + /** + * Existing fingerprint for comparison against event file. Can be null, then no comparison is performed. + */ + public final EventFingerprint compareFingerprint; - public static class EventFingerprint { - List timeArray = new ArrayList<>(); - public Map eventTypeCounter = new HashMap<>(); - Integer stringHash; + FloatListIterator iterator = null; - public EventFingerprint(List timeArray, Map eventTypeCounter, Integer stringHash) { - this.timeArray = timeArray; - this.eventTypeCounter = eventTypeCounter; - this.stringHash = stringHash; - } - public EventFingerprint() { - this.timeArray = timeArray; - this.eventTypeCounter = eventTypeCounter; - Hash stringHash; - } + /** + * Result of the comparison. + */ + public ComparisonResult comparisonResult; + public String comparisonMessage; - public List getTimeArray() { - return timeArray; - } + public FingerprintEventHandler() { + this.compareFingerprint = null; + } - public Map getEventTypeCounter() { - return eventTypeCounter; - } + public FingerprintEventHandler(EventFingerprint compareFingerprint) { + this.compareFingerprint = compareFingerprint; + this.comparisonResult = null; + } - public int getStringHash() { - return stringHash; - } + public EventFingerprint getEventFingerprint() { + return eventFingerprint; + } + + @Override + public void handleEvent(Event event) { + eventFingerprint.addTimeStamp(event.getTime()); + + String lexicographicSortedString = toLexicographicSortedString(event); - public void addHashCode(String stringToAdd) { - if (stringToAdd == null) { - return; - } - int hashToAdd = stringToAdd.hashCode(); - if (stringHash == null) { - stringHash = hashToAdd; - } else { - stringHash += hashToAdd; + + if (compareFingerprint != null) { + if(iterator== null){ + iterator = eventFingerprint.timeArray.iterator(); + } + if (this.comparisonResult == null) { + if(iterator.hasNext()){ + Float entry = iterator.nextFloat(); + if((float) event.getTime()!=entry){ + this.comparisonResult = ComparisonResult.DIFFERENT_TIMESTEPS; + this.comparisonMessage = "Difference occurred in this event "+lexicographicSortedString; + } + } } } - public void printFingerprint(FingerprintEventHandler.EventFingerprint fingerprint) { - System.out.println("Time Array:"); - var i = 0; - for (Float value : fingerprint.getTimeArray()) { - i++; - if(i % 100000 == 0) - System.out.println(value); - } + eventFingerprint.addEventType(event.getEventType()); - System.out.println("Event Type Counter:"); - for (Map.Entry entry : fingerprint.getEventTypeCounter().entrySet()) { - System.out.println(entry.getKey() + ": " + entry.getValue()); - } + eventFingerprint.addHashCode(lexicographicSortedString); + } + + /** + * :TODO ask about function renaming. Also I don't like result name: + *

+ * Finish processing of the events file and return comparison result (if compare fingerprint was present). + * If the result is not equal it will generate a {@link #comparisonMessage}. + */ + ComparisonResult finishProcessing() { - System.out.println("String Hash: " + fingerprint.getStringHash()); + byte[] hash = eventFingerprint.computeHash(); + + //That means that during event handling different timestampt occured + if (comparisonResult != null) { + return comparisonResult; } - } - public void addEventType(String str) { - // Increment the count for the given string - Map eventTypeCounter = eventFingerprint.getEventTypeCounter(); - eventTypeCounter.put(str, eventTypeCounter.getOrDefault(str, 0) + 1); - } + if (compareFingerprint == null) + return null; - public int getEventCount(String str) { - // Get the count for the given string - return eventFingerprint.getEventTypeCounter().getOrDefault(str, 0); - } + for (Object2IntMap.Entry entry1 : compareFingerprint.eventTypeCounter.object2IntEntrySet()) { + String key = entry1.getKey(); + int count1 = entry1.getIntValue(); + int count2 = eventFingerprint.eventTypeCounter.getInt(key); + if (count1 != count2) { + comparisonResult = (comparisonResult == null ? ComparisonResult.WRONG_EVENT_COUNT : comparisonResult); + comparisonMessage = (comparisonMessage == null ? "" : comparisonMessage) + ("\r\nCount for key '" + key + "' differs: " + count1 + " != " + count2); + } + } - @Override - public void handleEvent(Event event) { - addEventType(event.getEventType()); - eventFingerprint.getTimeArray().add((float) event.getTime()); - String lexicographicSortedString = toLexicographicSortedString(event); - eventFingerprint.addHashCode(lexicographicSortedString); + if (!Arrays.equals(hash, compareFingerprint.hash)) { + comparisonResult = ComparisonResult.MISSING_EVENT; + comparisonMessage = "Difference occured in this hash of 2 files"; + + //log.warn(comparisonMessage); + return comparisonResult; + } + + if (comparisonResult == null) { + comparisonResult = ComparisonResult.FILES_ARE_EQUAL; + comparisonMessage = "Seems to be equal :)"; + } + + //log.warn(comparisonMessage); + // TODO: compare and generate messages + return comparisonResult; } @@ -105,7 +127,7 @@ private String toLexicographicSortedString(Event event) { StringBuilder tmp = new StringBuilder(); final String key = e.getKey(); - // dont look at cerftain attributes + // don't look at certain attributes switch (key) { case Event.ATTRIBUTE_X: case Event.ATTRIBUTE_Y: @@ -128,67 +150,4 @@ private String toLexicographicSortedString(Event event) { eventStr.append(" | "); return eventStr.toString(); } - - - public static void writeEventFingerprintToFile(String filePath, EventFingerprint eventFingerprint) { - try (DataOutputStream dataOutputStream = new DataOutputStream(new FileOutputStream(filePath))) { - // Write header and version - dataOutputStream.writeByte(header); - dataOutputStream.writeByte(version); - - // Write time array size and elements - dataOutputStream.writeInt(eventFingerprint.getTimeArray().size()); - for (float time : eventFingerprint.getTimeArray()) { - dataOutputStream.writeFloat(time); - } - - // Write event type counter map size and elements - dataOutputStream.writeInt(eventFingerprint.getEventTypeCounter().size()); - for (Map.Entry entry : eventFingerprint.getEventTypeCounter().entrySet()) { - dataOutputStream.writeUTF(entry.getKey()); - dataOutputStream.writeInt(entry.getValue()); - } - - // Write string hash - dataOutputStream.writeInt(eventFingerprint.getStringHash()); - } catch (IOException e) { - e.printStackTrace(); - } - } - - public static EventFingerprint readEventFingerprintFromFile(String fingerprintPath) { - EventFingerprint eventFingerprint = null; - - try (DataInputStream dataInputStream = new DataInputStream(new FileInputStream(fingerprintPath))) { - // Read header and version - byte fileHeader = dataInputStream.readByte(); - byte fileVersion = dataInputStream.readByte(); - - // Read time array - int timeArraySize = dataInputStream.readInt(); - List timeArray = new ArrayList<>(); - for (int i = 0; i < timeArraySize; i++) { - timeArray.add(dataInputStream.readFloat()); - } - - // Read event type counter map - int eventTypeCounterSize = dataInputStream.readInt(); - Map eventTypeCounter = new HashMap<>(); - for (int i = 0; i < eventTypeCounterSize; i++) { - String eventType = dataInputStream.readUTF(); - int count = dataInputStream.readInt(); - eventTypeCounter.put(eventType, count); - } - - // Read string hash - int stringHash = dataInputStream.readInt(); - - // Create EventFingerprint object - eventFingerprint = new EventFingerprint(timeArray, eventTypeCounter, stringHash); - } catch (IOException e) { - e.printStackTrace(); - } - - return eventFingerprint; - } } diff --git a/matsim/src/test/java/org/matsim/examples/EquilTest.java b/matsim/src/test/java/org/matsim/examples/EquilTest.java index 96794010292..31f714c3d34 100644 --- a/matsim/src/test/java/org/matsim/examples/EquilTest.java +++ b/matsim/src/test/java/org/matsim/examples/EquilTest.java @@ -41,6 +41,7 @@ import org.matsim.core.scenario.ScenarioUtils; import org.matsim.testcases.MatsimTestUtils; import org.matsim.utils.eventsfilecomparison.EventsFileComparator; +import org.matsim.utils.eventsfilecomparison.ComparisonResult; public class EquilTest { private static final Logger log = LogManager.getLogger( EquilTest.class ) ; @@ -84,7 +85,7 @@ void testEquil(boolean isUsingFastCapacityUpdate) { writer.closeFile(); - final EventsFileComparator.Result result = new EventsFileComparator().setIgnoringCoordinates( true ).runComparison( referenceFileName , eventsFileName ); - Assertions.assertEquals(EventsFileComparator.Result.FILES_ARE_EQUAL, result, "different event files." ); + final ComparisonResult result = new EventsFileComparator().setIgnoringCoordinates( true ).runComparison( referenceFileName , eventsFileName ); + Assertions.assertEquals(ComparisonResult.FILES_ARE_EQUAL, result, "different event files." ); } } diff --git a/matsim/src/test/java/org/matsim/examples/OnePercentBerlin10sIT.java b/matsim/src/test/java/org/matsim/examples/OnePercentBerlin10sIT.java index 89237ec5b1a..f6d862c5785 100644 --- a/matsim/src/test/java/org/matsim/examples/OnePercentBerlin10sIT.java +++ b/matsim/src/test/java/org/matsim/examples/OnePercentBerlin10sIT.java @@ -41,6 +41,7 @@ import org.matsim.core.scenario.ScenarioUtils; import org.matsim.testcases.MatsimTestUtils; import org.matsim.utils.eventsfilecomparison.EventsFileComparator; +import org.matsim.utils.eventsfilecomparison.ComparisonResult; public class OnePercentBerlin10sIT { @@ -92,7 +93,7 @@ void testOnePercent10sQSim() { writer.closeFile(); System.out.println("reffile: " + referenceEventsFileName); - assertEquals( EventsFileComparator.Result.FILES_ARE_EQUAL, + assertEquals( ComparisonResult.FILES_ARE_EQUAL, new EventsFileComparator().setIgnoringCoordinates( true ).runComparison( referenceEventsFileName, eventsFileName ), "different event files" ); @@ -137,7 +138,7 @@ void testOnePercent10sQSimTryEndTimeThenDuration() { writer.closeFile(); - assertEquals( EventsFileComparator.Result.FILES_ARE_EQUAL, + assertEquals( ComparisonResult.FILES_ARE_EQUAL, new EventsFileComparator().setIgnoringCoordinates( true ).runComparison( referenceEventsFileName, eventsFileName ), "different event files" ); diff --git a/matsim/src/test/java/org/matsim/testcases/MatsimTestUtils.java b/matsim/src/test/java/org/matsim/testcases/MatsimTestUtils.java index 87696593899..112bcaaddaf 100644 --- a/matsim/src/test/java/org/matsim/testcases/MatsimTestUtils.java +++ b/matsim/src/test/java/org/matsim/testcases/MatsimTestUtils.java @@ -32,6 +32,7 @@ import org.matsim.core.utils.io.IOUtils; import org.matsim.core.utils.misc.CRCChecksum; import org.matsim.utils.eventsfilecomparison.EventsFileComparator; +import org.matsim.utils.eventsfilecomparison.ComparisonResult; import java.io.BufferedReader; import java.io.File; @@ -316,7 +317,7 @@ public static void assertEqualFilesLineByLine(String inputFilename, String outpu } public static void assertEqualEventsFiles( String filename1, String filename2 ) { - Assertions.assertEquals(EventsFileComparator.Result.FILES_ARE_EQUAL ,EventsFileComparator.compare(filename1, filename2) ); + Assertions.assertEquals(ComparisonResult.FILES_ARE_EQUAL ,EventsFileComparator.compare(filename1, filename2) ); } public static void assertEqualFilesBasedOnCRC( String filename1, String filename2 ) { diff --git a/matsim/src/test/java/org/matsim/utils/eventsfilecomparison/EventsFileComparatorTest.java b/matsim/src/test/java/org/matsim/utils/eventsfilecomparison/EventsFileComparatorTest.java index 6618b85dc17..9a52ac8593e 100644 --- a/matsim/src/test/java/org/matsim/utils/eventsfilecomparison/EventsFileComparatorTest.java +++ b/matsim/src/test/java/org/matsim/utils/eventsfilecomparison/EventsFileComparatorTest.java @@ -20,7 +20,7 @@ package org.matsim.utils.eventsfilecomparison; import static org.junit.jupiter.api.Assertions.assertEquals; -import static org.matsim.utils.eventsfilecomparison.EventsFileComparator.Result.*; +import static org.matsim.utils.eventsfilecomparison.ComparisonResult.*; import org.junit.jupiter.api.Test; import org.junit.jupiter.api.extension.RegisterExtension; diff --git a/matsim/src/test/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparatorTest.java b/matsim/src/test/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparatorTest.java new file mode 100644 index 00000000000..f4da125da35 --- /dev/null +++ b/matsim/src/test/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparatorTest.java @@ -0,0 +1,34 @@ +package org.matsim.utils.eventsfilecomparison; + +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; +import org.matsim.testcases.MatsimTestUtils; + +import static org.junit.jupiter.api.Assertions.*; + +class EventsFileFingerprintComparatorTest { + + @RegisterExtension + MatsimTestUtils utils = new MatsimTestUtils(); + + @Test + void testEqual() { + + utils.getClassInputDirectory(); + + + } + + @Test + void testDiffTimestamp() { + } + + @Test + void testDiffCounts() { + } + + @Test + void testDiffContent() { + } + +} From 0b49dbf973a6fd5eb0448d149f1e5d0681fb5f27 Mon Sep 17 00:00:00 2001 From: zlukich Date: Fri, 1 Mar 2024 10:47:05 +0100 Subject: [PATCH 05/12] added tests for comparison, added folders for input files --- .../EventsFileFingerprintComparator.java | 3 +- .../FingerprintEventHandler.java | 22 +++++++++--- .../EventsFileFingerprintComparatorTest.java | 33 ++++++++++++++++-- .../events_correct_fp.zst | Bin 0 -> 192 bytes .../output_events.diff-hash.xml | 14 ++++++++ .../output_events.diff-num-timestamps.xml | 12 +++++++ .../output_events.diff-type-count.xml | 14 ++++++++ .../output_events.xml | 14 ++++++++ 8 files changed, 104 insertions(+), 8 deletions(-) create mode 100644 matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/events_correct_fp.zst create mode 100644 matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.diff-hash.xml create mode 100644 matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.diff-num-timestamps.xml create mode 100644 matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.diff-type-count.xml create mode 100644 matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.xml diff --git a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparator.java b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparator.java index d6984cca371..8b46c0de6f3 100644 --- a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparator.java +++ b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparator.java @@ -19,6 +19,7 @@ public static ComparisonResult compare(final String fingerprint, final String ev EventFingerprint correctFingerprint = EventFingerprint.read(fingerprint); FingerprintEventHandler handler = new FingerprintEventHandler(correctFingerprint); + EventsManager manager = EventsUtils.createEventsManager(); manager.addHandler(handler); @@ -82,7 +83,7 @@ public static ComparisonResult compareFingerprints(final String fp1, final Strin log_message = String.format("Difference occurred hash codes hash of first file is %s, hash of second is %s", Arrays.toString(hash1), Arrays.toString(hash2)); log.warn(log_message); - return ComparisonResult.MISSING_EVENT; + return ComparisonResult.DIFFERENT_EVENT_ATTRIBUTES; } log_message = "Files are equal"; diff --git a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/FingerprintEventHandler.java b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/FingerprintEventHandler.java index 030d4b9bb82..65f05c7c52a 100644 --- a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/FingerprintEventHandler.java +++ b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/FingerprintEventHandler.java @@ -54,7 +54,7 @@ public void handleEvent(Event event) { if (compareFingerprint != null) { if(iterator== null){ - iterator = eventFingerprint.timeArray.iterator(); + this.iterator = compareFingerprint.timeArray.iterator(); } if (this.comparisonResult == null) { if(iterator.hasNext()){ @@ -73,7 +73,6 @@ public void handleEvent(Event event) { } /** - * :TODO ask about function renaming. Also I don't like result name: *

* Finish processing of the events file and return comparison result (if compare fingerprint was present). * If the result is not equal it will generate a {@link #comparisonMessage}. @@ -82,7 +81,9 @@ ComparisonResult finishProcessing() { byte[] hash = eventFingerprint.computeHash(); - //That means that during event handling different timestampt occured + /** Difference was found in {@link EventFingerprint#timeArray} + * + */ if (comparisonResult != null) { return comparisonResult; } @@ -90,6 +91,7 @@ ComparisonResult finishProcessing() { if (compareFingerprint == null) return null; + //Handling EventTypeCounter differences for (Object2IntMap.Entry entry1 : compareFingerprint.eventTypeCounter.object2IntEntrySet()) { String key = entry1.getKey(); int count1 = entry1.getIntValue(); @@ -100,15 +102,25 @@ ComparisonResult finishProcessing() { } } + /** Difference was found in {@link EventFingerprint#eventTypeCounter} + * + */ + if (comparisonResult != null) { + return comparisonResult; + } + if (!Arrays.equals(hash, compareFingerprint.hash)) { - comparisonResult = ComparisonResult.MISSING_EVENT; + comparisonResult = ComparisonResult.DIFFERENT_EVENT_ATTRIBUTES; comparisonMessage = "Difference occured in this hash of 2 files"; - //log.warn(comparisonMessage); + /** Difference was found in {@link EventFingerprint#hash} + * + */ return comparisonResult; } + if (comparisonResult == null) { comparisonResult = ComparisonResult.FILES_ARE_EQUAL; comparisonMessage = "Seems to be equal :)"; diff --git a/matsim/src/test/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparatorTest.java b/matsim/src/test/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparatorTest.java index f4da125da35..497a609ba17 100644 --- a/matsim/src/test/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparatorTest.java +++ b/matsim/src/test/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparatorTest.java @@ -2,8 +2,12 @@ import org.junit.jupiter.api.Test; import org.junit.jupiter.api.extension.RegisterExtension; +import org.matsim.core.events.EventsUtils; import org.matsim.testcases.MatsimTestUtils; +import java.io.File; + +import static org.assertj.core.api.Assertions.assertThat; import static org.junit.jupiter.api.Assertions.*; class EventsFileFingerprintComparatorTest { @@ -14,21 +18,46 @@ class EventsFileFingerprintComparatorTest { @Test void testEqual() { - utils.getClassInputDirectory(); + assertThat(EventsUtils.createAndCompareEventsFingerprint( + new File(utils.getClassInputDirectory(), "events_correct_fp.zst").toString(), + new File(utils.getClassInputDirectory(), "output_events.xml").toString() + )).isEqualTo(ComparisonResult.FILES_ARE_EQUAL); + + assertThat(EventsUtils.createAndCompareEventsFingerprint( + new File(utils.getClassInputDirectory(), "events_correct_fp.zst").toString(), + new File(utils.getClassInputDirectory(), "output_events.xml").toString() + )).isEqualTo(ComparisonResult.FILES_ARE_EQUAL); } @Test - void testDiffTimestamp() { + void testDiffTimesteps() { + + assertThat(EventsUtils.createAndCompareEventsFingerprint( + new File(utils.getClassInputDirectory(), "events_correct_fp.zst").toString(), + new File(utils.getClassInputDirectory(), "output_events.diff-num-timestamps.xml").toString() + )).isEqualTo(ComparisonResult.DIFFERENT_TIMESTEPS); + + } @Test void testDiffCounts() { + + assertThat(EventsUtils.createAndCompareEventsFingerprint( + new File(utils.getClassInputDirectory(), "events_correct_fp.zst").toString(), + new File(utils.getClassInputDirectory(), "output_events.diff-type-count.xml").toString() + )).isEqualTo(ComparisonResult.WRONG_EVENT_COUNT); + } @Test void testDiffContent() { + assertThat(EventsUtils.createAndCompareEventsFingerprint( + new File(utils.getClassInputDirectory(), "events_correct_fp.zst").toString(), + new File(utils.getClassInputDirectory(), "output_events.diff-hash.xml").toString() + )).isEqualTo(ComparisonResult.DIFFERENT_EVENT_ATTRIBUTES); } } diff --git a/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/events_correct_fp.zst b/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/events_correct_fp.zst new file mode 100644 index 0000000000000000000000000000000000000000..8dcc4bcb291d66982d59757a2c826f32cc181a9b GIT binary patch literal 192 zcmV;x06+gIwJ-euSfm92Vo4+}fK};W3d~A8ghreNdX5M&qv}!0u_RQ(13*tw$WxF{cqNu{q%6DvyBtw!7Nj5H&@e* zPGiIeF?*T3Ehlm_d9MW~x1Sc4JI%^jY`XM6ss@?WaKitLgwbKk+j6NR{x>4l0U(y* u6sSu4e>kox4FDuVD9D%?0l-7g_QHx+n4?8%?8|M-Yzw#r7h8680RRAE=v68J literal 0 HcmV?d00001 diff --git a/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.diff-hash.xml b/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.diff-hash.xml new file mode 100644 index 00000000000..e0cd3563911 --- /dev/null +++ b/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.diff-hash.xml @@ -0,0 +1,14 @@ + + + + + + + + + + + + + + \ No newline at end of file diff --git a/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.diff-num-timestamps.xml b/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.diff-num-timestamps.xml new file mode 100644 index 00000000000..422b58d651d --- /dev/null +++ b/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.diff-num-timestamps.xml @@ -0,0 +1,12 @@ + + + + + + + + + + + + \ No newline at end of file diff --git a/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.diff-type-count.xml b/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.diff-type-count.xml new file mode 100644 index 00000000000..31662fae9c0 --- /dev/null +++ b/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.diff-type-count.xml @@ -0,0 +1,14 @@ + + + + + + + + + + + + + + \ No newline at end of file diff --git a/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.xml b/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.xml new file mode 100644 index 00000000000..4b2bbbd1b92 --- /dev/null +++ b/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.xml @@ -0,0 +1,14 @@ + + + + + + + + + + + + + + \ No newline at end of file From 54c0db775d82f060e0acd277af320be7b16cc4b7 Mon Sep 17 00:00:00 2001 From: zlukich Date: Tue, 12 Mar 2024 16:54:35 +0100 Subject: [PATCH 06/12] added hash order logic, also javadoc, also fixed minor errors --- .../org/matsim/core/events/EventsUtils.java | 55 ++++++-- .../EventFingerprint.java | 5 +- .../EventsFileFingerprintComparator.java | 56 ++++---- .../FingerprintEventHandler.java | 133 +++++++++++------- .../EventsFileFingerprintComparatorTest.java | 105 ++++++++++++-- .../output_events.attribute-order.xml | 14 ++ .../output_events.diff-hash.xml | 2 +- .../output_events.event-order-wrong_logic.xml | 14 ++ .../output_events.event-order.xml | 14 ++ .../output_events.one-more-event.xml | 15 ++ 10 files changed, 304 insertions(+), 109 deletions(-) create mode 100644 matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.attribute-order.xml create mode 100644 matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.event-order-wrong_logic.xml create mode 100644 matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.event-order.xml create mode 100644 matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.one-more-event.xml diff --git a/matsim/src/main/java/org/matsim/core/events/EventsUtils.java b/matsim/src/main/java/org/matsim/core/events/EventsUtils.java index 2f02eb3c6a9..6788f429b9a 100644 --- a/matsim/src/main/java/org/matsim/core/events/EventsUtils.java +++ b/matsim/src/main/java/org/matsim/core/events/EventsUtils.java @@ -22,13 +22,20 @@ package org.matsim.core.events; import org.apache.commons.compress.utils.FileNameUtils; +import org.apache.logging.log4j.LogManager; +import org.apache.logging.log4j.Logger; import org.matsim.core.api.experimental.events.EventsManager; import org.matsim.core.config.Config; import org.matsim.core.controler.Injector; import org.matsim.utils.eventsfilecomparison.*; +import java.io.File; + public final class EventsUtils { + private static final Logger log = LogManager.getLogger(EventsUtils.class); + + /** * Create a events manager instance that guarantees causality of processed events across all handlers. */ @@ -69,15 +76,13 @@ public static EventsManager getParallelFeedableInstance(EventsManager events) { /** * Create and write fingerprint file for events. */ - public static void createEventsFingerprint(String eventFile, String outputFingerprintFile) { + public static FingerprintEventHandler createEventsFingerprint(String eventFile, String outputFingerprintFile) { + + FingerprintEventHandler handler = EventsFileFingerprintComparator.createFingerprintHandler(eventFile, null); - EventsManager manager = createEventsManager(); - FingerprintEventHandler fingerprintEventHandler = new FingerprintEventHandler(); - manager.addHandler(fingerprintEventHandler); - EventsUtils.readEvents(manager, eventFile); - manager.finishProcessing(); + EventFingerprint.write(outputFingerprintFile, handler.getEventFingerprint()); - EventFingerprint.write(outputFingerprintFile, fingerprintEventHandler.eventFingerprint); + return handler; } @@ -86,18 +91,38 @@ public static void createEventsFingerprint(String eventFile, String outputFinger * * @return comparison results */ - public static ComparisonResult createAndCompareEventsFingerprint(String inputFingerprint, String eventFile) { + public static ComparisonResult createAndCompareEventsFingerprint(String eventFile, String compareFingerprint) { + + + File file = new File(eventFile); + + String path = file.getPath().replace(".xml", ""); + + FingerprintEventHandler handler = EventsFileFingerprintComparator.createFingerprintHandler(eventFile, compareFingerprint); + EventFingerprint.write(path + ".fp.zst", handler.getEventFingerprint()); + + if (handler.getComparisonMessage() != null) + log.warn(handler.getComparisonMessage()); + + return handler.getComparisonResult(); + } + + public static void assertEqualEventsFingerprint(String eventFile, String compareFingerprint) { + + // TODO: execute event comparison + // if not equal, throw this error, with error message from com + + File file = new File(eventFile); + + String path = file.getPath().replace(".xml", ""); - // header byte, version byte - // bin array time stamps - // event type counter map - // one hash (sha1?) + FingerprintEventHandler handler = EventsFileFingerprintComparator.createFingerprintHandler(eventFile, compareFingerprint); + EventFingerprint.write(path + ".fp.zst", handler.getEventFingerprint()); - String baseName = FileNameUtils.getBaseName(eventFile).replace(".xml", ""); - createEventsFingerprint(eventFile, baseName + ".fp.zst"); + if (handler.getComparisonResult() != ComparisonResult.FILES_ARE_EQUAL) + throw new AssertionError(handler.getComparisonMessage()); - return EventsFileFingerprintComparator.compare(inputFingerprint, eventFile); } public static void readEvents(EventsManager events, String filename) { diff --git a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventFingerprint.java b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventFingerprint.java index 71ca4b5697d..da36054eda9 100644 --- a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventFingerprint.java +++ b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventFingerprint.java @@ -107,7 +107,10 @@ public static void write(String filePath, EventFingerprint eventFingerprint) { } // Write byte hash - dataOutputStream.write(eventFingerprint.computeHash()); + assert !Arrays.equals(eventFingerprint.hash, new byte[20]): "Hash was not computed"; + + dataOutputStream.write(eventFingerprint.hash); + } catch (IOException e) { e.printStackTrace(); } diff --git a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparator.java b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparator.java index 8b46c0de6f3..332ad985df6 100644 --- a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparator.java +++ b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparator.java @@ -1,24 +1,29 @@ package org.matsim.utils.eventsfilecomparison; import it.unimi.dsi.fastutil.objects.Object2IntMap; -import it.unimi.dsi.fastutil.objects.ObjectIterator; import org.apache.logging.log4j.LogManager; import org.apache.logging.log4j.Logger; import org.matsim.core.api.experimental.events.EventsManager; import org.matsim.core.events.EventsUtils; -import java.util.*; +import javax.annotation.Nullable; +import java.util.Arrays; -public class EventsFileFingerprintComparator { +public final class EventsFileFingerprintComparator { private static final Logger log = LogManager.getLogger(EventsFileComparator.class); + private EventsFileFingerprintComparator() { + + } - public static ComparisonResult compare(final String fingerprint, final String eventsfile) { - EventFingerprint correctFingerprint = EventFingerprint.read(fingerprint); + /** + * Create and compare event fingerprints and return the handler holding resulting information. + */ + public static FingerprintEventHandler createFingerprintHandler(final String eventsfile, @Nullable String compareFingerprint) { - FingerprintEventHandler handler = new FingerprintEventHandler(correctFingerprint); + FingerprintEventHandler handler = new FingerprintEventHandler(compareFingerprint != null ? EventFingerprint.read(compareFingerprint) : null); EventsManager manager = EventsUtils.createEventsManager(); @@ -27,33 +32,28 @@ public static ComparisonResult compare(final String fingerprint, final String ev EventsUtils.readEvents(manager, eventsfile); manager.finishProcessing(); + handler.finishProcessing(); - ComparisonResult result = handler.finishProcessing(); - - log.warn(handler.comparisonMessage); - - // All fields are equal - return result; + return handler; } public static ComparisonResult compareFingerprints(final String fp1, final String fp2) { - EventFingerprint fingerprint1 = EventFingerprint.read(fp1); EventFingerprint fingerprint2 = EventFingerprint.read(fp2); - String log_message = ""; + String logMessage = ""; //Check if time array size is the same - if(fingerprint1.timeArray.size()!=fingerprint2.timeArray.size()){ - log_message = "Different number of timesteps"; - log.warn(log_message); + if (fingerprint1.timeArray.size() != fingerprint2.timeArray.size()) { + logMessage = "Different number of timesteps"; + log.warn(logMessage); return ComparisonResult.DIFFERENT_NUMBER_OF_TIMESTEPS; } //Check if both time arrays have the same timesteps - if(!Arrays.equals(fingerprint1.timeArray.toFloatArray(),fingerprint2.timeArray.toFloatArray())){ - log_message = "Different timesteps"; - log.warn(log_message); + if (!Arrays.equals(fingerprint1.timeArray.toFloatArray(), fingerprint2.timeArray.toFloatArray())) { + logMessage = "Different timesteps"; + log.warn(logMessage); return ComparisonResult.DIFFERENT_TIMESTEPS; } @@ -66,11 +66,11 @@ public static ComparisonResult compareFingerprints(final String fp1, final Strin int count2 = fingerprint2.eventTypeCounter.getInt(key); if (count1 != count2) { countDiffers = true; - log_message = log_message + ("\r\nCount for key '" + key + "' differs: " + count1 + " != " + count2); + logMessage += "\r\nCount for event type '%s' differs: %d != %d".formatted(key, count1, count2); } } - if(countDiffers){ - log.warn(log_message); + if (countDiffers) { + log.warn(logMessage); return ComparisonResult.WRONG_EVENT_COUNT; } @@ -78,18 +78,14 @@ public static ComparisonResult compareFingerprints(final String fp1, final Strin //Check if total hash is the same byte[] hash1 = fingerprint1.hash; byte[] hash2 = fingerprint2.hash; - if(!Arrays.equals(hash1, hash2)){ + if (!Arrays.equals(hash1, hash2)) { - log_message = String.format("Difference occurred hash codes hash of first file is %s, hash of second is %s", Arrays.toString(hash1), Arrays.toString(hash2)); + logMessage = String.format("Difference occurred hash codes hash of first file is %s, hash of second is %s", Arrays.toString(hash1), Arrays.toString(hash2)); - log.warn(log_message); + log.warn(logMessage); return ComparisonResult.DIFFERENT_EVENT_ATTRIBUTES; } - log_message = "Files are equal"; - log.warn(log_message); - - // TODO: compare and generate messages return ComparisonResult.FILES_ARE_EQUAL; } diff --git a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/FingerprintEventHandler.java b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/FingerprintEventHandler.java index 65f05c7c52a..71c85e4641b 100644 --- a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/FingerprintEventHandler.java +++ b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/FingerprintEventHandler.java @@ -2,34 +2,43 @@ import it.unimi.dsi.fastutil.floats.FloatListIterator; import it.unimi.dsi.fastutil.objects.Object2IntMap; -import it.unimi.dsi.fastutil.objects.ObjectIterator; import org.matsim.api.core.v01.events.Event; import org.matsim.core.events.handler.BasicEventHandler; -import org.matsim.counts.algorithms.graphs.helper.Comp; import java.util.*; +/** + * Handler for creating and comparing {@link EventFingerprint}. + */ public final class FingerprintEventHandler implements BasicEventHandler { - /** * Generated finger print. */ - public final EventFingerprint eventFingerprint = new EventFingerprint(); + private final EventFingerprint eventFingerprint = new EventFingerprint(); /** - * Existing fingerprint for comparison against event file. Can be null, then no comparison is performed. + * Precision for timestamp comparison. */ - public final EventFingerprint compareFingerprint; + private static final float EPS = 1e-8f; - FloatListIterator iterator = null; + /** + * Accumulate all event strings for same timestamps. + */ + private final List hashAccumulationList = new ArrayList<>(); + /** + * Existing fingerprint for comparison against event file. Can be null, then no comparison is performed. + */ + private final EventFingerprint compareFingerprint; + + private FloatListIterator iterator = null; /** * Result of the comparison. */ - public ComparisonResult comparisonResult; - public String comparisonMessage; + private ComparisonResult comparisonResult; + private String comparisonMessage; public FingerprintEventHandler() { this.compareFingerprint = null; @@ -44,32 +53,73 @@ public EventFingerprint getEventFingerprint() { return eventFingerprint; } + public ComparisonResult getComparisonResult() { + return comparisonResult; + } + + public String getComparisonMessage() { + return comparisonMessage; + } + @Override public void handleEvent(Event event) { - eventFingerprint.addTimeStamp(event.getTime()); - - String lexicographicSortedString = toLexicographicSortedString(event); + String lexicographicSortedString = toLexicographicSortedString(event); if (compareFingerprint != null) { - if(iterator== null){ + if (iterator == null) { this.iterator = compareFingerprint.timeArray.iterator(); } + if (this.comparisonResult == null) { - if(iterator.hasNext()){ - Float entry = iterator.nextFloat(); - if((float) event.getTime()!=entry){ + if (iterator.hasNext()) { + float entry = iterator.nextFloat(); + //Comparing floats with precision + if (Math.abs((float) event.getTime() - entry) >= EPS) { this.comparisonResult = ComparisonResult.DIFFERENT_TIMESTEPS; - this.comparisonMessage = "Difference occurred in this event "+lexicographicSortedString; + this.comparisonMessage = "Difference occurred in this event time=" + event.getTime() + lexicographicSortedString; } + } else { + this.comparisonResult = ComparisonResult.DIFFERENT_TIMESTEPS; + this.comparisonMessage = "Additional event time=" + event.getTime() + lexicographicSortedString; } } } eventFingerprint.addEventType(event.getEventType()); - eventFingerprint.addHashCode(lexicographicSortedString); + + //First timestep, nothing to accumulate + if (eventFingerprint.timeArray.isEmpty()) { + hashAccumulationList.add(lexicographicSortedString); + } else { + float lastTime = eventFingerprint.timeArray.getFloat(eventFingerprint.timeArray.size() - 1); + //If new time is the same as previous, add to accumulation list, event hash calculation is not ready + if (lastTime == event.getTime()) { + hashAccumulationList.add(lexicographicSortedString); + } + //if new time differs from previous, all hash can be calculated + else { + accumulateHash(); + hashAccumulationList.add(lexicographicSortedString); + } + } + + eventFingerprint.addTimeStamp(event.getTime()); + + //eventFingerprint.addHashCode(lexicographicSortedString); + } + + private void accumulateHash() { + + Collections.sort(hashAccumulationList); + + for (String str : hashAccumulationList) { + eventFingerprint.addHashCode(str); + } + + hashAccumulationList.clear(); } /** @@ -77,19 +127,18 @@ public void handleEvent(Event event) { * Finish processing of the events file and return comparison result (if compare fingerprint was present). * If the result is not equal it will generate a {@link #comparisonMessage}. */ - ComparisonResult finishProcessing() { + void finishProcessing() { + + if (!hashAccumulationList.isEmpty()) { + accumulateHash(); + } byte[] hash = eventFingerprint.computeHash(); - /** Difference was found in {@link EventFingerprint#timeArray} - * - */ - if (comparisonResult != null) { - return comparisonResult; - } + //hash = eventFingerprint.computeHash(); if (compareFingerprint == null) - return null; + return; //Handling EventTypeCounter differences for (Object2IntMap.Entry entry1 : compareFingerprint.eventTypeCounter.object2IntEntrySet()) { @@ -97,40 +146,26 @@ ComparisonResult finishProcessing() { int count1 = entry1.getIntValue(); int count2 = eventFingerprint.eventTypeCounter.getInt(key); if (count1 != count2) { + comparisonMessage = comparisonMessage == null ? "" : comparisonMessage; + comparisonResult = (comparisonResult == null ? ComparisonResult.WRONG_EVENT_COUNT : comparisonResult); - comparisonMessage = (comparisonMessage == null ? "" : comparisonMessage) + ("\r\nCount for key '" + key + "' differs: " + count1 + " != " + count2); + comparisonMessage += "\nCount for event type '%s' differs: %d (in fingerprint) != %d (in events)".formatted(key, count1, count2); } } - /** Difference was found in {@link EventFingerprint#eventTypeCounter} - * - */ + // Difference was found in {@link EventFingerprint#eventTypeCounter} if (comparisonResult != null) { - return comparisonResult; + return; } - + // only check hash if there was no difference up until here if (!Arrays.equals(hash, compareFingerprint.hash)) { comparisonResult = ComparisonResult.DIFFERENT_EVENT_ATTRIBUTES; - comparisonMessage = "Difference occured in this hash of 2 files"; - - /** Difference was found in {@link EventFingerprint#hash} - * - */ - return comparisonResult; + comparisonMessage = "Difference occurred in this hash of 2 files"; + return; } - - if (comparisonResult == null) { - comparisonResult = ComparisonResult.FILES_ARE_EQUAL; - comparisonMessage = "Seems to be equal :)"; - } - - //log.warn(comparisonMessage); - - // TODO: compare and generate messages - return comparisonResult; - + comparisonResult = ComparisonResult.FILES_ARE_EQUAL; } private String toLexicographicSortedString(Event event) { diff --git a/matsim/src/test/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparatorTest.java b/matsim/src/test/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparatorTest.java index 497a609ba17..7c6b1ce589b 100644 --- a/matsim/src/test/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparatorTest.java +++ b/matsim/src/test/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparatorTest.java @@ -15,19 +15,15 @@ class EventsFileFingerprintComparatorTest { @RegisterExtension MatsimTestUtils utils = new MatsimTestUtils(); + @Test void testEqual() { - assertThat(EventsUtils.createAndCompareEventsFingerprint( - new File(utils.getClassInputDirectory(), "events_correct_fp.zst").toString(), - new File(utils.getClassInputDirectory(), "output_events.xml").toString() + new File(utils.getClassInputDirectory(), "output_events.xml").toString(), + new File(utils.getClassInputDirectory(), "output_events.correct.fp.zst").toString() )).isEqualTo(ComparisonResult.FILES_ARE_EQUAL); - assertThat(EventsUtils.createAndCompareEventsFingerprint( - new File(utils.getClassInputDirectory(), "events_correct_fp.zst").toString(), - new File(utils.getClassInputDirectory(), "output_events.xml").toString() - )).isEqualTo(ComparisonResult.FILES_ARE_EQUAL); } @@ -35,8 +31,8 @@ void testEqual() { void testDiffTimesteps() { assertThat(EventsUtils.createAndCompareEventsFingerprint( - new File(utils.getClassInputDirectory(), "events_correct_fp.zst").toString(), - new File(utils.getClassInputDirectory(), "output_events.diff-num-timestamps.xml").toString() + new File(utils.getClassInputDirectory(), "output_events.diff-num-timestamps.xml").toString(), + new File(utils.getClassInputDirectory(), "output_events.correct.fp.zst").toString() )).isEqualTo(ComparisonResult.DIFFERENT_TIMESTEPS); @@ -46,8 +42,8 @@ void testDiffTimesteps() { void testDiffCounts() { assertThat(EventsUtils.createAndCompareEventsFingerprint( - new File(utils.getClassInputDirectory(), "events_correct_fp.zst").toString(), - new File(utils.getClassInputDirectory(), "output_events.diff-type-count.xml").toString() + new File(utils.getClassInputDirectory(), "output_events.diff-type-count.xml").toString(), + new File(utils.getClassInputDirectory(), "output_events.correct.fp.zst").toString() )).isEqualTo(ComparisonResult.WRONG_EVENT_COUNT); } @@ -55,9 +51,92 @@ void testDiffCounts() { @Test void testDiffContent() { assertThat(EventsUtils.createAndCompareEventsFingerprint( - new File(utils.getClassInputDirectory(), "events_correct_fp.zst").toString(), - new File(utils.getClassInputDirectory(), "output_events.diff-hash.xml").toString() + new File(utils.getClassInputDirectory(), "output_events.diff-hash.xml").toString(), + new File(utils.getClassInputDirectory(), "output_events.correct.fp.zst").toString() )).isEqualTo(ComparisonResult.DIFFERENT_EVENT_ATTRIBUTES); } + @Test + void testAdditionalEvent() { + assertThat(EventsUtils.createAndCompareEventsFingerprint( + new File(utils.getClassInputDirectory(), "output_events.one-more-event.xml").toString(), + new File(utils.getClassInputDirectory(), "output_events.correct.fp.zst").toString() + )).isEqualTo(ComparisonResult.DIFFERENT_TIMESTEPS); + } + + @Test + void testAttributeOrder() { + assertThat(EventsUtils.createAndCompareEventsFingerprint( + new File(utils.getClassInputDirectory(), "output_events.attribute-order.xml").toString(), + new File(utils.getClassInputDirectory(), "output_events.correct.fp.zst").toString() + )).isEqualTo(ComparisonResult.FILES_ARE_EQUAL); + } + + @Test + void testEventOrder() { + assertThat(EventsUtils.compareEventsFiles( + new File(utils.getClassInputDirectory(), "output_events.xml").toString(), + new File(utils.getClassInputDirectory(), "output_events.event-order-wrong_logic.xml").toString() + )).isEqualTo(ComparisonResult.FILES_ARE_EQUAL); + + assertThat(EventsUtils.createAndCompareEventsFingerprint( + new File(utils.getClassInputDirectory(), "output_events.event-order-wrong_logic.xml").toString(), + new File(utils.getClassInputDirectory(), "output_events.correct.fp.zst").toString() + )).isEqualTo(ComparisonResult.FILES_ARE_EQUAL); + + } + + + // TODO: add tests comparing only fingerprints + + @Test + void testEqualFingerprints() { + + EventsUtils.createEventsFingerprint(new File(utils.getClassInputDirectory(), "output_events.xml").toString(), new File(utils.getClassInputDirectory(), "output_events.fp.zst").toString()); + + assertThat(EventsFileFingerprintComparator.compareFingerprints( + new File(utils.getClassInputDirectory(), "output_events.fp.zst").toString(), + new File(utils.getClassInputDirectory(), "output_events.correct.fp.zst").toString() + )).isEqualTo(ComparisonResult.FILES_ARE_EQUAL); + + + } + + @Test + void testDiffTimestepsFingerprints() { + + assertThat(EventsFileFingerprintComparator.compareFingerprints( + new File(utils.getClassInputDirectory(), "output_events.diff-num-timestamps.fp.zst").toString(), + new File(utils.getClassInputDirectory(), "output_events.correct.fp.zst").toString() + )).isEqualTo(ComparisonResult.DIFFERENT_NUMBER_OF_TIMESTEPS); + + + } + + @Test + void testDiffCountsFingerprints() { + + assertThat(EventsFileFingerprintComparator.compareFingerprints( + new File(utils.getClassInputDirectory(), "output_events.diff-type-count.fp.zst").toString(), + new File(utils.getClassInputDirectory(), "output_events.correct.fp.zst").toString() + )).isEqualTo(ComparisonResult.WRONG_EVENT_COUNT); + + } + + @Test + void assertion() { + + AssertionError err = assertThrows(AssertionError.class, () -> { + EventsUtils.assertEqualEventsFingerprint( + new File(utils.getClassInputDirectory(), "output_events.diff-num-timestamps.xml").toString(), + new File(utils.getClassInputDirectory(), "output_events.correct.fp.zst").toString() + ); + }); + + assertThat(err).message().isEqualTo(""" + Difference occurred in this event time=48067.0 | link=1 | type=entered link | vehicle=5 |\s + Count for event type 'entered link' differs: 2 (in fingerprint) != 1 (in events) + Count for event type 'vehicle leaves traffic' differs: 1 (in fingerprint) != 0 (in events)"""); + + } } diff --git a/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.attribute-order.xml b/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.attribute-order.xml new file mode 100644 index 00000000000..6885fa8af5d --- /dev/null +++ b/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.attribute-order.xml @@ -0,0 +1,14 @@ + + + + + + + + + + + + + + \ No newline at end of file diff --git a/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.diff-hash.xml b/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.diff-hash.xml index e0cd3563911..4ae06c2369f 100644 --- a/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.diff-hash.xml +++ b/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.diff-hash.xml @@ -7,7 +7,7 @@ - + diff --git a/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.event-order-wrong_logic.xml b/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.event-order-wrong_logic.xml new file mode 100644 index 00000000000..1112364be0c --- /dev/null +++ b/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.event-order-wrong_logic.xml @@ -0,0 +1,14 @@ + + + + + + + + + + + + + + \ No newline at end of file diff --git a/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.event-order.xml b/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.event-order.xml new file mode 100644 index 00000000000..c4b9bbf33de --- /dev/null +++ b/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.event-order.xml @@ -0,0 +1,14 @@ + + + + + + + + + + + + + + \ No newline at end of file diff --git a/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.one-more-event.xml b/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.one-more-event.xml new file mode 100644 index 00000000000..663754db63f --- /dev/null +++ b/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.one-more-event.xml @@ -0,0 +1,15 @@ + + + + + + + + + + + + + + + \ No newline at end of file From b43b125514917c9f5981de0f1cb5b11a6352f859 Mon Sep 17 00:00:00 2001 From: rakow Date: Wed, 13 Mar 2024 10:50:41 +0100 Subject: [PATCH 07/12] cleaned up tests, improved API --- matsim/.gitignore | 2 +- .../org/matsim/core/events/EventsUtils.java | 39 +++--- .../EventFingerprint.java | 112 ++++++++++-------- .../EventsFileFingerprintComparator.java | 4 +- .../EventsFileFingerprintComparatorTest.java | 57 ++++----- .../correct.fp.zst | Bin 0 -> 196 bytes ...e-order.xml => events.attribute-order.xml} | 0 ...nts.diff-hash.xml => events.diff-hash.xml} | 0 ...mps.xml => events.diff-num-timestamps.xml} | 0 .../events.diff-type-count.fp.zst | Bin 0 -> 194 bytes ...e-count.xml => events.diff-type-count.xml} | 0 ...xml => events.event-order-wrong_logic.xml} | 0 ...event-order.xml => events.event-order.xml} | 0 ...re-event.xml => events.one-more-event.xml} | 0 .../{output_events.xml => events_correct.xml} | 0 .../events_correct_fp.zst | Bin 192 -> 0 bytes 16 files changed, 116 insertions(+), 98 deletions(-) create mode 100644 matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/correct.fp.zst rename matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/{output_events.attribute-order.xml => events.attribute-order.xml} (100%) rename matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/{output_events.diff-hash.xml => events.diff-hash.xml} (100%) rename matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/{output_events.diff-num-timestamps.xml => events.diff-num-timestamps.xml} (100%) create mode 100644 matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/events.diff-type-count.fp.zst rename matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/{output_events.diff-type-count.xml => events.diff-type-count.xml} (100%) rename matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/{output_events.event-order-wrong_logic.xml => events.event-order-wrong_logic.xml} (100%) rename matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/{output_events.event-order.xml => events.event-order.xml} (100%) rename matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/{output_events.one-more-event.xml => events.one-more-event.xml} (100%) rename matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/{output_events.xml => events_correct.xml} (100%) delete mode 100644 matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/events_correct_fp.zst diff --git a/matsim/.gitignore b/matsim/.gitignore index 029e60cc779..a35b8f59ee1 100644 --- a/matsim/.gitignore +++ b/matsim/.gitignore @@ -3,7 +3,6 @@ test/output bin .settings -bin target output out @@ -11,3 +10,4 @@ src/main/java/.gitignore /output_fastCapacityUpdate_false/ /output_fastCapacityUpdate_true/ /nullevents.xml.gz +*.zst \ No newline at end of file diff --git a/matsim/src/main/java/org/matsim/core/events/EventsUtils.java b/matsim/src/main/java/org/matsim/core/events/EventsUtils.java index 6788f429b9a..7601a9091ad 100644 --- a/matsim/src/main/java/org/matsim/core/events/EventsUtils.java +++ b/matsim/src/main/java/org/matsim/core/events/EventsUtils.java @@ -21,7 +21,6 @@ package org.matsim.core.events; -import org.apache.commons.compress.utils.FileNameUtils; import org.apache.logging.log4j.LogManager; import org.apache.logging.log4j.Logger; import org.matsim.core.api.experimental.events.EventsManager; @@ -56,6 +55,11 @@ public static EventsManager createEventsManager(Config config) { return events; } + public static void readEvents(EventsManager events, String filename) { + new MatsimEventsReader(events).readFile(filename); + } + + /** * The SimStepParallelEventsManagerImpl can handle events from multiple threads. * The (Parallel)EventsMangerImpl cannot, therefore it has to be wrapped into a @@ -89,16 +93,16 @@ public static FingerprintEventHandler createEventsFingerprint(String eventFile, /** * Compares existing event file against fingerprint file. This will also create new fingerprint file along the input events. * + * @param eventFile local events file + * @param compareFingerprintFile path or uri to fingerprint file + * * @return comparison results */ - public static ComparisonResult createAndCompareEventsFingerprint(String eventFile, String compareFingerprint) { - + public static ComparisonResult createAndCompareEventsFingerprint(File eventFile, String compareFingerprintFile) { - File file = new File(eventFile); + String path = eventFile.getPath().replaceFirst("\\.xml[.a-zA-z0-9]*$", ""); - String path = file.getPath().replace(".xml", ""); - - FingerprintEventHandler handler = EventsFileFingerprintComparator.createFingerprintHandler(eventFile, compareFingerprint); + FingerprintEventHandler handler = EventsFileFingerprintComparator.createFingerprintHandler(eventFile.toString(), compareFingerprintFile); EventFingerprint.write(path + ".fp.zst", handler.getEventFingerprint()); if (handler.getComparisonMessage() != null) @@ -107,16 +111,17 @@ public static ComparisonResult createAndCompareEventsFingerprint(String eventFil return handler.getComparisonResult(); } - public static void assertEqualEventsFingerprint(String eventFile, String compareFingerprint) { - - // TODO: execute event comparison - // if not equal, throw this error, with error message from com - - File file = new File(eventFile); + /** + * Compares existing event file against fingerprint file. This will also create new fingerprint file along the input events. + * + * @throws AssertionError if comparison fails + * @see #createAndCompareEventsFingerprint(File, String) + */ + public static void assertEqualEventsFingerprint(File eventFile, String compareFingerprintFile) { - String path = file.getPath().replace(".xml", ""); + String path = eventFile.getPath().replaceFirst("\\.xml[.a-zA-z0-9]*$", ""); - FingerprintEventHandler handler = EventsFileFingerprintComparator.createFingerprintHandler(eventFile, compareFingerprint); + FingerprintEventHandler handler = EventsFileFingerprintComparator.createFingerprintHandler(eventFile.toString(), compareFingerprintFile); EventFingerprint.write(path + ".fp.zst", handler.getEventFingerprint()); @@ -125,10 +130,6 @@ public static void assertEqualEventsFingerprint(String eventFile, String compare } - public static void readEvents(EventsManager events, String filename) { - new MatsimEventsReader(events).readFile(filename); - } - public static ComparisonResult compareEventsFiles(String filename1, String filename2) { return EventsFileComparator.compare(filename1, filename2); } diff --git a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventFingerprint.java b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventFingerprint.java index da36054eda9..c2c0a8610a5 100644 --- a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventFingerprint.java +++ b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventFingerprint.java @@ -6,12 +6,27 @@ import it.unimi.dsi.fastutil.objects.Object2IntOpenHashMap; import org.matsim.core.utils.io.IOUtils; -import java.io.*; +import java.io.DataInputStream; +import java.io.DataOutputStream; +import java.io.IOException; +import java.io.UncheckedIOException; import java.nio.charset.StandardCharsets; import java.security.MessageDigest; import java.security.NoSuchAlgorithmException; -import java.util.*; - +import java.util.Arrays; +import java.util.Map; + +/** + * Class holding reduced information about an events file. + * If two such fingerprint are different, one can conclude that the event files are semantically different. + *

+ * The fingerprint is based on the following information: + * - Array of all timestamps + * - Counts of each event type + * - Hash of string concatenation of all event strings + * + * Note: Events with the same timestamp are allowed to occur in any order. + */ public final class EventFingerprint { /** @@ -35,7 +50,7 @@ private EventFingerprint(FloatList timeArray, Object2IntMap eventTypeCou this.digest = null; } - public EventFingerprint() { + EventFingerprint() { this.timeArray = new FloatArrayList(); this.eventTypeCounter = new Object2IntOpenHashMap<>(); this.hash = new byte[20]; @@ -47,47 +62,6 @@ public EventFingerprint() { } } - public void addTimeStamp(double timestamp) { - timeArray.add((float) timestamp); - } - - public void addEventType(String str) { - // Increment the count for the given string - eventTypeCounter.mergeInt(str, 1, Integer::sum); - } - - public void addHashCode(String stringToAdd) { - if (stringToAdd == null) { - return; - } - - digest.update(stringToAdd.getBytes(StandardCharsets.UTF_8)); - } - - byte[] computeHash() { - if (this.digest == null) - throw new IllegalStateException("Hash was from from input and can not be computed"); - - byte[] digest = this.digest.digest(); - System.arraycopy(digest, 0, hash, 0, hash.length); - return hash; - } - - public static void printFingerprint(EventFingerprint fingerprint) { - System.out.println("Time Array:"); - var i = 0; - for (Float value : fingerprint.timeArray) { - System.out.print(value); - } - - System.out.println("Event Type Counter:"); - for (Map.Entry entry : fingerprint.eventTypeCounter.entrySet()) { - System.out.println(entry.getKey() + ": " + entry.getValue()); - } - - System.out.println("String Hash: " + Arrays.toString(fingerprint.hash)); - } - public static void write(String filePath, EventFingerprint eventFingerprint) { try (DataOutputStream dataOutputStream = new DataOutputStream(IOUtils.getOutputStream(IOUtils.getFileUrl(filePath), false))) { // Write header and version @@ -106,23 +80,28 @@ public static void write(String filePath, EventFingerprint eventFingerprint) { dataOutputStream.writeInt(entry.getValue()); } - // Write byte hash - assert !Arrays.equals(eventFingerprint.hash, new byte[20]): "Hash was not computed"; + // Hash should always be computed at this point + assert !Arrays.equals(eventFingerprint.hash, new byte[20]) : "Hash was not computed"; + // Write byte hash dataOutputStream.write(eventFingerprint.hash); } catch (IOException e) { - e.printStackTrace(); + throw new UncheckedIOException(e); } } public static EventFingerprint read(String fingerprintPath) { - EventFingerprint eventFingerprint = null; + EventFingerprint eventFingerprint; try (DataInputStream dataInputStream = new DataInputStream(IOUtils.getInputStream(IOUtils.getFileUrl(fingerprintPath)))) { // Read header and version int fileHeader = dataInputStream.readInt(); + if (fileHeader != EventFingerprint.HEADER_V1) { + throw new IllegalArgumentException("Invalid fingerprint file header"); + } + // Read time array int timeArraySize = dataInputStream.readInt(); FloatList timeArray = new FloatArrayList(); @@ -150,4 +129,39 @@ public static EventFingerprint read(String fingerprintPath) { return eventFingerprint; } + + void addTimeStamp(double timestamp) { + timeArray.add((float) timestamp); + } + + void addEventType(String str) { + // Increment the count for the given string + eventTypeCounter.mergeInt(str, 1, Integer::sum); + } + + void addHashCode(String stringToAdd) { + if (stringToAdd == null) { + return; + } + + digest.update(stringToAdd.getBytes(StandardCharsets.UTF_8)); + } + + byte[] computeHash() { + if (this.digest == null) + throw new IllegalStateException("Hash was from from input and can not be computed"); + + byte[] digest = this.digest.digest(); + System.arraycopy(digest, 0, hash, 0, hash.length); + return hash; + } + + @Override + public String toString() { + return "EventFingerprint{" + + "timeArray=" + timeArray.size() + + ", eventTypeCounter=" + eventTypeCounter + + ", hash=" + Arrays.toString(hash) + + '}'; + } } diff --git a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparator.java b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparator.java index 332ad985df6..3f0447d91cf 100644 --- a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparator.java +++ b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparator.java @@ -9,12 +9,14 @@ import javax.annotation.Nullable; import java.util.Arrays; +/** + * Utility class for comparing events and fingerprints. + */ public final class EventsFileFingerprintComparator { private static final Logger log = LogManager.getLogger(EventsFileComparator.class); private EventsFileFingerprintComparator() { - } diff --git a/matsim/src/test/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparatorTest.java b/matsim/src/test/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparatorTest.java index 7c6b1ce589b..a14100b52d1 100644 --- a/matsim/src/test/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparatorTest.java +++ b/matsim/src/test/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparatorTest.java @@ -15,15 +15,18 @@ class EventsFileFingerprintComparatorTest { @RegisterExtension MatsimTestUtils utils = new MatsimTestUtils(); - @Test void testEqual() { assertThat(EventsUtils.createAndCompareEventsFingerprint( - new File(utils.getClassInputDirectory(), "output_events.xml").toString(), - new File(utils.getClassInputDirectory(), "output_events.correct.fp.zst").toString() + new File(utils.getClassInputDirectory(), "events_correct.xml"), + new File(utils.getClassInputDirectory(), "correct.fp.zst").toString() )).isEqualTo(ComparisonResult.FILES_ARE_EQUAL); + // Check if file has been created + assertThat(new File(utils.getClassInputDirectory(), "events_correct.fp.zst")) + .exists() + .size().isGreaterThan(0); } @@ -31,8 +34,8 @@ void testEqual() { void testDiffTimesteps() { assertThat(EventsUtils.createAndCompareEventsFingerprint( - new File(utils.getClassInputDirectory(), "output_events.diff-num-timestamps.xml").toString(), - new File(utils.getClassInputDirectory(), "output_events.correct.fp.zst").toString() + new File(utils.getClassInputDirectory(), "events.diff-num-timestamps.xml"), + new File(utils.getClassInputDirectory(), "correct.fp.zst").toString() )).isEqualTo(ComparisonResult.DIFFERENT_TIMESTEPS); @@ -42,8 +45,8 @@ void testDiffTimesteps() { void testDiffCounts() { assertThat(EventsUtils.createAndCompareEventsFingerprint( - new File(utils.getClassInputDirectory(), "output_events.diff-type-count.xml").toString(), - new File(utils.getClassInputDirectory(), "output_events.correct.fp.zst").toString() + new File(utils.getClassInputDirectory(), "events.diff-type-count.xml"), + new File(utils.getClassInputDirectory(), "correct.fp.zst").toString() )).isEqualTo(ComparisonResult.WRONG_EVENT_COUNT); } @@ -51,52 +54,50 @@ void testDiffCounts() { @Test void testDiffContent() { assertThat(EventsUtils.createAndCompareEventsFingerprint( - new File(utils.getClassInputDirectory(), "output_events.diff-hash.xml").toString(), - new File(utils.getClassInputDirectory(), "output_events.correct.fp.zst").toString() + new File(utils.getClassInputDirectory(), "events.diff-hash.xml"), + new File(utils.getClassInputDirectory(), "correct.fp.zst").toString() )).isEqualTo(ComparisonResult.DIFFERENT_EVENT_ATTRIBUTES); } @Test void testAdditionalEvent() { assertThat(EventsUtils.createAndCompareEventsFingerprint( - new File(utils.getClassInputDirectory(), "output_events.one-more-event.xml").toString(), - new File(utils.getClassInputDirectory(), "output_events.correct.fp.zst").toString() + new File(utils.getClassInputDirectory(), "events.one-more-event.xml"), + new File(utils.getClassInputDirectory(), "correct.fp.zst").toString() )).isEqualTo(ComparisonResult.DIFFERENT_TIMESTEPS); } @Test void testAttributeOrder() { assertThat(EventsUtils.createAndCompareEventsFingerprint( - new File(utils.getClassInputDirectory(), "output_events.attribute-order.xml").toString(), - new File(utils.getClassInputDirectory(), "output_events.correct.fp.zst").toString() + new File(utils.getClassInputDirectory(), "events.attribute-order.xml"), + new File(utils.getClassInputDirectory(), "correct.fp.zst").toString() )).isEqualTo(ComparisonResult.FILES_ARE_EQUAL); } @Test void testEventOrder() { assertThat(EventsUtils.compareEventsFiles( - new File(utils.getClassInputDirectory(), "output_events.xml").toString(), - new File(utils.getClassInputDirectory(), "output_events.event-order-wrong_logic.xml").toString() + new File(utils.getClassInputDirectory(), "events_correct.xml").toString(), + new File(utils.getClassInputDirectory(), "events.event-order-wrong_logic.xml").toString() )).isEqualTo(ComparisonResult.FILES_ARE_EQUAL); assertThat(EventsUtils.createAndCompareEventsFingerprint( - new File(utils.getClassInputDirectory(), "output_events.event-order-wrong_logic.xml").toString(), - new File(utils.getClassInputDirectory(), "output_events.correct.fp.zst").toString() + new File(utils.getClassInputDirectory(), "events.event-order-wrong_logic.xml"), + new File(utils.getClassInputDirectory(), "correct.fp.zst").toString() )).isEqualTo(ComparisonResult.FILES_ARE_EQUAL); } - // TODO: add tests comparing only fingerprints - @Test void testEqualFingerprints() { - EventsUtils.createEventsFingerprint(new File(utils.getClassInputDirectory(), "output_events.xml").toString(), new File(utils.getClassInputDirectory(), "output_events.fp.zst").toString()); + EventsUtils.createEventsFingerprint(new File(utils.getClassInputDirectory(), "events_correct.xml").toString(), new File(utils.getClassInputDirectory(), "events.fp.zst").toString()); assertThat(EventsFileFingerprintComparator.compareFingerprints( - new File(utils.getClassInputDirectory(), "output_events.fp.zst").toString(), - new File(utils.getClassInputDirectory(), "output_events.correct.fp.zst").toString() + new File(utils.getClassInputDirectory(), "events.fp.zst").toString(), + new File(utils.getClassInputDirectory(), "correct.fp.zst").toString() )).isEqualTo(ComparisonResult.FILES_ARE_EQUAL); @@ -106,8 +107,8 @@ void testEqualFingerprints() { void testDiffTimestepsFingerprints() { assertThat(EventsFileFingerprintComparator.compareFingerprints( - new File(utils.getClassInputDirectory(), "output_events.diff-num-timestamps.fp.zst").toString(), - new File(utils.getClassInputDirectory(), "output_events.correct.fp.zst").toString() + new File(utils.getClassInputDirectory(), "events.diff-num-timestamps.fp.zst").toString(), + new File(utils.getClassInputDirectory(), "correct.fp.zst").toString() )).isEqualTo(ComparisonResult.DIFFERENT_NUMBER_OF_TIMESTEPS); @@ -117,8 +118,8 @@ void testDiffTimestepsFingerprints() { void testDiffCountsFingerprints() { assertThat(EventsFileFingerprintComparator.compareFingerprints( - new File(utils.getClassInputDirectory(), "output_events.diff-type-count.fp.zst").toString(), - new File(utils.getClassInputDirectory(), "output_events.correct.fp.zst").toString() + new File(utils.getClassInputDirectory(), "events.diff-type-count.fp.zst").toString(), + new File(utils.getClassInputDirectory(), "correct.fp.zst").toString() )).isEqualTo(ComparisonResult.WRONG_EVENT_COUNT); } @@ -128,8 +129,8 @@ void assertion() { AssertionError err = assertThrows(AssertionError.class, () -> { EventsUtils.assertEqualEventsFingerprint( - new File(utils.getClassInputDirectory(), "output_events.diff-num-timestamps.xml").toString(), - new File(utils.getClassInputDirectory(), "output_events.correct.fp.zst").toString() + new File(utils.getClassInputDirectory(), "events.diff-num-timestamps.xml"), + new File(utils.getClassInputDirectory(), "correct.fp.zst").toString() ); }); diff --git a/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/correct.fp.zst b/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/correct.fp.zst new file mode 100644 index 0000000000000000000000000000000000000000..7bf62ca48e26975c0b7ed02c29daada8427db755 GIT binary patch literal 196 zcmdPcs{fZE;s`543a49uz99nx1Gn1>9tJ*l>%$E0wpu_w7XwdfUP)?EYKlTmW?nW> zgo%MYv8X7sEHMYjVPxP;Ni9e$Dk&{WWlKyhNzF?U4oEF3&d+lN=_(FO&B#p75i3*3 zNlh$EEmkNgN=!@3O!nc-NlhzZWZ(d5DJ}tO2N|WY!Q$}xlNm+}6d&AQ7IO5A#^z7U oc^OokHBQWsU_9V_{@bCh@EMC;v)-J}J)Oh7gH=4b3}^%c07D-^;{X5v literal 0 HcmV?d00001 diff --git a/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.attribute-order.xml b/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/events.attribute-order.xml similarity index 100% rename from matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.attribute-order.xml rename to matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/events.attribute-order.xml diff --git a/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.diff-hash.xml b/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/events.diff-hash.xml similarity index 100% rename from matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.diff-hash.xml rename to matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/events.diff-hash.xml diff --git a/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.diff-num-timestamps.xml b/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/events.diff-num-timestamps.xml similarity index 100% rename from matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.diff-num-timestamps.xml rename to matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/events.diff-num-timestamps.xml diff --git a/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/events.diff-type-count.fp.zst b/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/events.diff-type-count.fp.zst new file mode 100644 index 0000000000000000000000000000000000000000..6437b813ac0fd615f3a74dea6c4492d7f5cca149 GIT binary patch literal 194 zcmdPcs{fZEVhbyS2&Y?sz99nx1Gn1>9tJ*l>%$E0wpu_w7XwdfUP)?EYKlTmW?nW> zgpq+gv8X7sEHQ^OCAA>2sHC(gl`S#3BsDKZI3TsCI6u!7q@g%0H6t@QN32XCCpEDw zwOFB~C^0Q9GuekTCpE2viGc&CrMLvB9b{yoZfn9FuFS{f&vgpd+TJXxYGnS-$Drb@ nabku9;{oUM-wt(!&sgl5_2zW$OkUpLf(JY?x literal 0 HcmV?d00001 diff --git a/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.diff-type-count.xml b/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/events.diff-type-count.xml similarity index 100% rename from matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.diff-type-count.xml rename to matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/events.diff-type-count.xml diff --git a/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.event-order-wrong_logic.xml b/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/events.event-order-wrong_logic.xml similarity index 100% rename from matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.event-order-wrong_logic.xml rename to matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/events.event-order-wrong_logic.xml diff --git a/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.event-order.xml b/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/events.event-order.xml similarity index 100% rename from matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.event-order.xml rename to matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/events.event-order.xml diff --git a/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.one-more-event.xml b/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/events.one-more-event.xml similarity index 100% rename from matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.one-more-event.xml rename to matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/events.one-more-event.xml diff --git a/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.xml b/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/events_correct.xml similarity index 100% rename from matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/output_events.xml rename to matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/events_correct.xml diff --git a/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/events_correct_fp.zst b/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/events_correct_fp.zst deleted file mode 100644 index 8dcc4bcb291d66982d59757a2c826f32cc181a9b..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 192 zcmV;x06+gIwJ-euSfm92Vo4+}fK};W3d~A8ghreNdX5M&qv}!0u_RQ(13*tw$WxF{cqNu{q%6DvyBtw!7Nj5H&@e* zPGiIeF?*T3Ehlm_d9MW~x1Sc4JI%^jY`XM6ss@?WaKitLgwbKk+j6NR{x>4l0U(y* u6sSu4e>kox4FDuVD9D%?0l-7g_QHx+n4?8%?8|M-Yzw#r7h8680RRAE=v68J From 463b5866b4a5faacf1a34770b8994040fb2e93a4 Mon Sep 17 00:00:00 2001 From: rakow Date: Wed, 13 Mar 2024 10:56:40 +0100 Subject: [PATCH 08/12] remove extra new line from error messages --- .../EventsFileFingerprintComparator.java | 6 +++++- .../utils/eventsfilecomparison/FingerprintEventHandler.java | 6 +++++- 2 files changed, 10 insertions(+), 2 deletions(-) diff --git a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparator.java b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparator.java index 3f0447d91cf..ca10bc0bf08 100644 --- a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparator.java +++ b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparator.java @@ -68,7 +68,11 @@ public static ComparisonResult compareFingerprints(final String fp1, final Strin int count2 = fingerprint2.eventTypeCounter.getInt(key); if (count1 != count2) { countDiffers = true; - logMessage += "\r\nCount for event type '%s' differs: %d != %d".formatted(key, count1, count2); + + if (!logMessage.isEmpty()) + logMessage += "\n"; + + logMessage += "Count for event type '%s' differs: %d != %d".formatted(key, count1, count2); } } if (countDiffers) { diff --git a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/FingerprintEventHandler.java b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/FingerprintEventHandler.java index 71c85e4641b..2469f65759e 100644 --- a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/FingerprintEventHandler.java +++ b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/FingerprintEventHandler.java @@ -149,7 +149,11 @@ void finishProcessing() { comparisonMessage = comparisonMessage == null ? "" : comparisonMessage; comparisonResult = (comparisonResult == null ? ComparisonResult.WRONG_EVENT_COUNT : comparisonResult); - comparisonMessage += "\nCount for event type '%s' differs: %d (in fingerprint) != %d (in events)".formatted(key, count1, count2); + + if (!comparisonMessage.isEmpty()) + comparisonMessage += "\n"; + + comparisonMessage += "Count for event type '%s' differs: %d (in fingerprint) != %d (in events)".formatted(key, count1, count2); } } From 1c91ca6618b3c04c6b570b055c74a3ca218fc785 Mon Sep 17 00:00:00 2001 From: rakow Date: Wed, 13 Mar 2024 11:10:57 +0100 Subject: [PATCH 09/12] update test --- .../RunGenerateSmallScaleCommercialTrafficTest.java | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/contribs/small-scale-traffic-generation/src/test/java/org/matsim/smallScaleCommercialTrafficGeneration/RunGenerateSmallScaleCommercialTrafficTest.java b/contribs/small-scale-traffic-generation/src/test/java/org/matsim/smallScaleCommercialTrafficGeneration/RunGenerateSmallScaleCommercialTrafficTest.java index d88d2318fd1..b5e92503851 100644 --- a/contribs/small-scale-traffic-generation/src/test/java/org/matsim/smallScaleCommercialTrafficGeneration/RunGenerateSmallScaleCommercialTrafficTest.java +++ b/contribs/small-scale-traffic-generation/src/test/java/org/matsim/smallScaleCommercialTrafficGeneration/RunGenerateSmallScaleCommercialTrafficTest.java @@ -34,7 +34,7 @@ import org.matsim.freight.carriers.CarriersUtils; import org.matsim.freight.carriers.FreightCarriersConfigGroup; import org.matsim.testcases.MatsimTestUtils; -import org.matsim.utils.eventsfilecomparison.EventsFileComparator; +import org.matsim.utils.eventsfilecomparison.ComparisonResult; import java.io.File; import java.util.Objects; @@ -119,7 +119,7 @@ void testMainRunAndResults() { // compare events String expected = utils.getPackageInputDirectory() + "test.output_events.xml.gz" ; String actual = utils.getOutputDirectory() + "test.output_events.xml.gz" ; - EventsFileComparator.Result result = EventsUtils.compareEventsFiles( expected, actual ); - Assertions.assertEquals( EventsFileComparator.Result.FILES_ARE_EQUAL, result ); + ComparisonResult result = EventsUtils.compareEventsFiles( expected, actual ); + Assertions.assertEquals( ComparisonResult.FILES_ARE_EQUAL, result ); } } From a6ca470c2fb2252503062b0406dbc86caf2e5dd0 Mon Sep 17 00:00:00 2001 From: rakow Date: Wed, 13 Mar 2024 11:15:34 +0100 Subject: [PATCH 10/12] add deprecation message to old enum --- .../utils/eventsfilecomparison/EventsFileComparator.java | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileComparator.java b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileComparator.java index 90c8d6e4e0c..0b759f11e95 100644 --- a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileComparator.java +++ b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileComparator.java @@ -193,4 +193,12 @@ private void setExitCode(final ComparisonResult errCode) { } } + /** + * Don't use this enum. See deprecation message. + * @deprecated Use {@link ComparisonResult} instead. This enum is not used anymore and empty now. + */ + @Deprecated + public enum Result { + } + } From d955a391ec74891b310295a5d01c5db4e0497f86 Mon Sep 17 00:00:00 2001 From: rakow Date: Wed, 13 Mar 2024 12:14:53 +0100 Subject: [PATCH 11/12] handle file errors as separate comparison result --- .../ComparisonResult.java | 2 +- .../EventFingerprint.java | 4 +-- .../EventsFileFingerprintComparator.java | 34 ++++++++++++++++--- .../FingerprintEventHandler.java | 8 +++++ 4 files changed, 40 insertions(+), 8 deletions(-) diff --git a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/ComparisonResult.java b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/ComparisonResult.java index c77394fddd4..b5e10fa2d65 100644 --- a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/ComparisonResult.java +++ b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/ComparisonResult.java @@ -3,4 +3,4 @@ /** * Result of event file comparison. */ -public enum ComparisonResult {FILES_ARE_EQUAL, DIFFERENT_NUMBER_OF_TIMESTEPS, DIFFERENT_TIMESTEPS, DIFFERENT_EVENT_ATTRIBUTES, MISSING_EVENT, WRONG_EVENT_COUNT} +public enum ComparisonResult {FILES_ARE_EQUAL, DIFFERENT_NUMBER_OF_TIMESTEPS, DIFFERENT_TIMESTEPS, DIFFERENT_EVENT_ATTRIBUTES, MISSING_EVENT, WRONG_EVENT_COUNT, FILE_ERROR} diff --git a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventFingerprint.java b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventFingerprint.java index c2c0a8610a5..74f254f746d 100644 --- a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventFingerprint.java +++ b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventFingerprint.java @@ -91,7 +91,7 @@ public static void write(String filePath, EventFingerprint eventFingerprint) { } } - public static EventFingerprint read(String fingerprintPath) { + public static EventFingerprint read(String fingerprintPath) throws IOException { EventFingerprint eventFingerprint; try (DataInputStream dataInputStream = new DataInputStream(IOUtils.getInputStream(IOUtils.getFileUrl(fingerprintPath)))) { @@ -123,8 +123,6 @@ public static EventFingerprint read(String fingerprintPath) { // Create EventFingerprint object eventFingerprint = new EventFingerprint(timeArray, eventTypeCounter, hash); - } catch (IOException e) { - throw new UncheckedIOException(e); } return eventFingerprint; diff --git a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparator.java b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparator.java index ca10bc0bf08..2152758b215 100644 --- a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparator.java +++ b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparator.java @@ -7,6 +7,8 @@ import org.matsim.core.events.EventsUtils; import javax.annotation.Nullable; +import java.io.IOException; +import java.io.UncheckedIOException; import java.util.Arrays; /** @@ -25,7 +27,19 @@ private EventsFileFingerprintComparator() { */ public static FingerprintEventHandler createFingerprintHandler(final String eventsfile, @Nullable String compareFingerprint) { - FingerprintEventHandler handler = new FingerprintEventHandler(compareFingerprint != null ? EventFingerprint.read(compareFingerprint) : null); + EventFingerprint fp = null; + Exception err = null; + if (compareFingerprint != null) { + try { + fp = EventFingerprint.read(compareFingerprint); + } catch (Exception e) { + log.warn("Could not read compare fingerprint from file: {}", compareFingerprint, e); + fp = new EventFingerprint(); + err = e; + } + } + + FingerprintEventHandler handler = new FingerprintEventHandler(fp); EventsManager manager = EventsUtils.createEventsManager(); @@ -36,15 +50,27 @@ public static FingerprintEventHandler createFingerprintHandler(final String even manager.finishProcessing(); handler.finishProcessing(); + // File error overwrite any other error + if (err != null) { + handler.setComparisonResult(ComparisonResult.FILE_ERROR); + handler.setComparisonMessage(err.getMessage()); + } + return handler; } public static ComparisonResult compareFingerprints(final String fp1, final String fp2) { - EventFingerprint fingerprint1 = EventFingerprint.read(fp1); - EventFingerprint fingerprint2 = EventFingerprint.read(fp2); + EventFingerprint fingerprint1; + EventFingerprint fingerprint2; + try { + fingerprint1 = EventFingerprint.read(fp1); + fingerprint2 = EventFingerprint.read(fp2); + } catch (IOException e) { + throw new UncheckedIOException(e); + } - String logMessage = ""; + String logMessage = ""; //Check if time array size is the same if (fingerprint1.timeArray.size() != fingerprint2.timeArray.size()) { logMessage = "Different number of timesteps"; diff --git a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/FingerprintEventHandler.java b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/FingerprintEventHandler.java index 2469f65759e..8e12c10a53c 100644 --- a/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/FingerprintEventHandler.java +++ b/matsim/src/main/java/org/matsim/utils/eventsfilecomparison/FingerprintEventHandler.java @@ -57,10 +57,18 @@ public ComparisonResult getComparisonResult() { return comparisonResult; } + void setComparisonResult(ComparisonResult comparisonResult) { + this.comparisonResult = comparisonResult; + } + public String getComparisonMessage() { return comparisonMessage; } + void setComparisonMessage(String comparisonMessage) { + this.comparisonMessage = comparisonMessage; + } + @Override public void handleEvent(Event event) { From 6ef8a300404d1f14a141e9becb5fd4cc3642b2e2 Mon Sep 17 00:00:00 2001 From: rakow Date: Mon, 18 Mar 2024 10:32:25 +0100 Subject: [PATCH 12/12] fix capital I in folder name --- .../correct.fp.zst | Bin .../events.attribute-order.xml | 0 .../events.diff-hash.xml | 0 .../events.diff-num-timestamps.xml | 0 .../events.diff-type-count.fp.zst | Bin .../events.diff-type-count.xml | 0 .../events.event-order-wrong_logic.xml | 0 .../events.event-order.xml | 0 .../events.one-more-event.xml | 0 .../events_correct.xml | 0 10 files changed, 0 insertions(+), 0 deletions(-) rename matsim/test/input/org/matsim/utils/eventsfilecomparison/{EventsFileFIngerprintComparatorTest => EventsFileFingerprintComparatorTest}/correct.fp.zst (100%) rename matsim/test/input/org/matsim/utils/eventsfilecomparison/{EventsFileFIngerprintComparatorTest => EventsFileFingerprintComparatorTest}/events.attribute-order.xml (100%) rename matsim/test/input/org/matsim/utils/eventsfilecomparison/{EventsFileFIngerprintComparatorTest => EventsFileFingerprintComparatorTest}/events.diff-hash.xml (100%) rename matsim/test/input/org/matsim/utils/eventsfilecomparison/{EventsFileFIngerprintComparatorTest => EventsFileFingerprintComparatorTest}/events.diff-num-timestamps.xml (100%) rename matsim/test/input/org/matsim/utils/eventsfilecomparison/{EventsFileFIngerprintComparatorTest => EventsFileFingerprintComparatorTest}/events.diff-type-count.fp.zst (100%) rename matsim/test/input/org/matsim/utils/eventsfilecomparison/{EventsFileFIngerprintComparatorTest => EventsFileFingerprintComparatorTest}/events.diff-type-count.xml (100%) rename matsim/test/input/org/matsim/utils/eventsfilecomparison/{EventsFileFIngerprintComparatorTest => EventsFileFingerprintComparatorTest}/events.event-order-wrong_logic.xml (100%) rename matsim/test/input/org/matsim/utils/eventsfilecomparison/{EventsFileFIngerprintComparatorTest => EventsFileFingerprintComparatorTest}/events.event-order.xml (100%) rename matsim/test/input/org/matsim/utils/eventsfilecomparison/{EventsFileFIngerprintComparatorTest => EventsFileFingerprintComparatorTest}/events.one-more-event.xml (100%) rename matsim/test/input/org/matsim/utils/eventsfilecomparison/{EventsFileFIngerprintComparatorTest => EventsFileFingerprintComparatorTest}/events_correct.xml (100%) diff --git a/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/correct.fp.zst b/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparatorTest/correct.fp.zst similarity index 100% rename from matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/correct.fp.zst rename to matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparatorTest/correct.fp.zst diff --git a/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/events.attribute-order.xml b/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparatorTest/events.attribute-order.xml similarity index 100% rename from matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/events.attribute-order.xml rename to matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparatorTest/events.attribute-order.xml diff --git a/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/events.diff-hash.xml b/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparatorTest/events.diff-hash.xml similarity index 100% rename from matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/events.diff-hash.xml rename to matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparatorTest/events.diff-hash.xml diff --git a/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/events.diff-num-timestamps.xml b/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparatorTest/events.diff-num-timestamps.xml similarity index 100% rename from matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/events.diff-num-timestamps.xml rename to matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparatorTest/events.diff-num-timestamps.xml diff --git a/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/events.diff-type-count.fp.zst b/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparatorTest/events.diff-type-count.fp.zst similarity index 100% rename from matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/events.diff-type-count.fp.zst rename to matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparatorTest/events.diff-type-count.fp.zst diff --git a/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/events.diff-type-count.xml b/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparatorTest/events.diff-type-count.xml similarity index 100% rename from matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/events.diff-type-count.xml rename to matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparatorTest/events.diff-type-count.xml diff --git a/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/events.event-order-wrong_logic.xml b/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparatorTest/events.event-order-wrong_logic.xml similarity index 100% rename from matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/events.event-order-wrong_logic.xml rename to matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparatorTest/events.event-order-wrong_logic.xml diff --git a/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/events.event-order.xml b/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparatorTest/events.event-order.xml similarity index 100% rename from matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/events.event-order.xml rename to matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparatorTest/events.event-order.xml diff --git a/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/events.one-more-event.xml b/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparatorTest/events.one-more-event.xml similarity index 100% rename from matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/events.one-more-event.xml rename to matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparatorTest/events.one-more-event.xml diff --git a/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/events_correct.xml b/matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparatorTest/events_correct.xml similarity index 100% rename from matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFIngerprintComparatorTest/events_correct.xml rename to matsim/test/input/org/matsim/utils/eventsfilecomparison/EventsFileFingerprintComparatorTest/events_correct.xml