diff --git a/CHANGELOG.rst b/CHANGELOG.rst index bdc7292bf84a2a62d055ce07d65a123cd8c8a4a0..0491189dca7e5d5acaa88eeff21589f99e9c84ee 100644 --- a/CHANGELOG.rst +++ b/CHANGELOG.rst @@ -3,6 +3,10 @@ Unreleased changes Version 0 --------- +0.9.0 / 2020-03-03 +~~~~~~~~~~~~~~~~~~ +* Added support for the ``usr`` field of events + 0.8.3 / 2020-02-25 ~~~~~~~~~~~~~~~~~~ * The times of snapshot and triggered hits were parsed as big endian (standard) diff --git a/README.rst b/README.rst index 543db640d5403c6217db48db73af9acf40bc869a..d4fc462a748efaa30bf8190dc5a3d7d487ee12ef 100644 --- a/README.rst +++ b/README.rst @@ -66,6 +66,8 @@ Tutorial * `reading events data <#reading-events-data>`__ + * `reading usr data of events <#reading-usr-data-of-events>`__ + * `reading hits data <#reading-hits-data>`__ * `reading tracks data <#reading-tracks-data>`__ @@ -75,6 +77,7 @@ Tutorial * `reading mc tracks data <#reading-mc-tracks-data>`__ + Introduction ------------ @@ -628,6 +631,44 @@ or the number of hits: 176 +reading usr data of events +"""""""""""""""""""""""""" + +To access the ``usr`` data of events, use the ``.usr`` property which behaves +like a dictionary and returns ``lazyarray``, compatible to the ``numpy.array`` +interface. The available keys can be accessed either as attributes or via a +dictionary lookup: + +.. code-block:: python3 + + >>> import km3io + >>> f = km3io.OfflineReader("tests/samples/usr-sample.root") + >>> f.usr + <km3io.offline.Usr at 0x7efd53a41eb0> + >>> print(f.usr) + RecoQuality: [85.45957235835593 68.74744265572737 50.18704013646688] + RecoNDF: [37.0 37.0 29.0] + CoC: [118.6302815337638 44.33580521344907 99.93916717621543] + ToT: [825.0 781.0 318.0] + ChargeAbove: [176.0 278.0 53.0] + ChargeBelow: [649.0 503.0 265.0] + ChargeRatio: [0.21333333333333335 0.3559539052496799 0.16666666666666666] + DeltaPosZ: [37.51967774166617 -10.280346193553832 13.67595659707355] + FirstPartPosZ: [135.29499707179326 41.46665612378939 107.39596803432326] + LastPartPosZ: [97.77531933012709 51.747002317343224 93.72001143724971] + NSnapHits: [51.0 107.0 98.0] + NTrigHits: [30.0 32.0 14.0] + NTrigDOMs: [7.0 11.0 7.0] + NTrigLines: [6.0 5.0 4.0] + NSpeedVetoHits: [0.0 0.0 0.0] + NGeometryVetoHits: [0.0 0.0 0.0] + ClassficationScore: [0.16863382173469108 0.17944356593281038 0.08155750660727408] + >>> f.usr.DeltaPosZ + <ChunkedArray [37.51967774166617 -10.280346193553832 13.67595659707355] at 0x7efd54013eb0> + >>> f.usr['RecoQuality'] + <ChunkedArray [85.45957235835593 68.74744265572737 50.18704013646688] at 0x7efd54034b50> + + reading hits data """"""""""""""""" diff --git a/km3io/offline.py b/km3io/offline.py index 7da446d3f6621c6debfca95544d777d461b71f1a..53c1a1a254e9afaa591fd9b682ba20079684c535 100644 --- a/km3io/offline.py +++ b/km3io/offline.py @@ -365,6 +365,7 @@ class OfflineReader: self._keys = None self._best_reco = None self._header = None + self._usr = None def __getitem__(self, item): return OfflineReader(file_path=self._file_path, data=self._data[item]) @@ -474,6 +475,12 @@ class OfflineReader: fitparameters=self.keys.fitparameters) return self._mc_tracks + @property + def usr(self): + if self._usr is None: + self._usr = Usr(self._file_path) + return self._usr + @property def best_reco(self): """returns the best reconstructed track fit data. The best fit is defined @@ -816,6 +823,44 @@ class OfflineReader: yield i, j +class Usr: + """Helper class to access AAObject usr stuff""" + def __init__(self, filepath): + self._f = uproot.open(filepath) + # Here, we assume that every event has the same names in the same order + # to massively increase the performance. This needs triple check if it's + # always the case; the usr-format is simply a very bad design. + try: + self._usr_names = [ + n.decode("utf-8") + for n in self._f['E']['Evt']['usr_names'].array()[0] + ] + except (KeyError, IndexError): # e.g. old aanet files + self._usr_names = [] + else: + self._usr_idx_lookup = { + name: index + for index, name in enumerate(self._usr_names) + } + self._usr_data = self._f['E']['Evt']['usr'].lazyarray( + basketcache=uproot.cache.ThreadSafeArrayCache( + BASKET_CACHE_SIZE)) + for name in self._usr_names: + setattr(self, name, self[name]) + + def __getitem__(self, item): + return self._usr_data[:, self._usr_idx_lookup[item]] + + def keys(self): + return self._usr_names + + def __str__(self): + entries = [] + for name in self.keys(): + entries.append("{}: {}".format(name, self[name])) + return '\n'.join(entries) + + class OfflineEvents: """wrapper for offline events""" def __init__(self, keys, values): diff --git a/tests/samples/usr-sample.root b/tests/samples/usr-sample.root new file mode 100644 index 0000000000000000000000000000000000000000..dbfb1bd69913b6e99adfee2bd00170260d1c625a Binary files /dev/null and b/tests/samples/usr-sample.root differ diff --git a/tests/test_offline.py b/tests/test_offline.py index 74ea43557734afd7033a2ca333e4dd40972adfba..f0ac1ba03b190547a063468d0bd3d53c0ec6d431 100644 --- a/tests/test_offline.py +++ b/tests/test_offline.py @@ -7,6 +7,7 @@ from km3io import OfflineReader SAMPLES_DIR = Path(__file__).parent / 'samples' OFFLINE_FILE = SAMPLES_DIR / 'aanet_v2.0.0.root' +OFFLINE_USR = SAMPLES_DIR / 'usr-sample.root' OFFLINE_NUMUCC = SAMPLES_DIR / "numucc.root" # with mc data @@ -478,3 +479,40 @@ class TestOfflineTrack(unittest.TestCase): def test_str(self): self.assertEqual(repr(self.track).split('\n\t')[0], 'offline track:') self.assertTrue("JGANDALF_LAMBDA" in repr(self.track)) + + +class TestUsr(unittest.TestCase): + def setUp(self): + self.f = OfflineReader(OFFLINE_USR) + + def test_str(self): + print(self.f.usr) + + def test_nonexistent_usr(self): + f = OfflineReader(SAMPLES_DIR / "daq_v1.0.0.root") + self.assertListEqual([], f.usr.keys()) + + def test_keys(self): + self.assertListEqual([ + 'RecoQuality', 'RecoNDF', 'CoC', 'ToT', 'ChargeAbove', + 'ChargeBelow', 'ChargeRatio', 'DeltaPosZ', 'FirstPartPosZ', + 'LastPartPosZ', 'NSnapHits', 'NTrigHits', 'NTrigDOMs', + 'NTrigLines', 'NSpeedVetoHits', 'NGeometryVetoHits', + 'ClassficationScore' + ], self.f.usr.keys()) + + def test_getitem(self): + assert np.allclose( + [118.6302815337638, 44.33580521344907, 99.93916717621543], + self.f.usr['CoC']) + assert np.allclose( + [37.51967774166617, -10.280346193553832, 13.67595659707355], + self.f.usr['DeltaPosZ']) + + def test_attributes(self): + assert np.allclose( + [118.6302815337638, 44.33580521344907, 99.93916717621543], + self.f.usr.CoC) + assert np.allclose( + [37.51967774166617, -10.280346193553832, 13.67595659707355], + self.f.usr.DeltaPosZ)