From 2236ade99f77fad597fd1205bb2c8fa5b369a970 Mon Sep 17 00:00:00 2001 From: "Brett M. Morris" Date: Fri, 21 Jun 2024 11:59:45 -0400 Subject: [PATCH] Prevent duplicate sub-interval data labels (#120) * temporary maxpin for numpy<2.0 * only append mission interval to data label when no label given --- CHANGES.rst | 2 ++ lcviz/conftest.py | 3 ++ lcviz/parsers.py | 46 ++++++++++++++++++------------ lcviz/tests/test_parser.py | 11 +++++++ lcviz/tests/test_plugin_markers.py | 4 +-- pyproject.toml | 1 + 6 files changed, 46 insertions(+), 21 deletions(-) diff --git a/CHANGES.rst b/CHANGES.rst index 48ed04c2..db14cb21 100644 --- a/CHANGES.rst +++ b/CHANGES.rst @@ -1,6 +1,8 @@ 0.5.0 (unreleased) ------------------ +* Prevent duplicate sub-intervals (quarter/sector/campaign) in data labels. [#120] + 0.4.1 (unreleased) ------------------ diff --git a/lcviz/conftest.py b/lcviz/conftest.py index 6d616903..767828cb 100644 --- a/lcviz/conftest.py +++ b/lcviz/conftest.py @@ -37,6 +37,9 @@ def light_curve_like_kepler_quarter(seed=42): ) lc['flux_alt'] = flux + 1 lc['flux_alt_err'] = flux_err + lc.meta['MISSION'] = 'KEPLER' + lc.meta['QUARTER'] = 10 + return lc diff --git a/lcviz/parsers.py b/lcviz/parsers.py index 4c32b875..06ea0ec6 100644 --- a/lcviz/parsers.py +++ b/lcviz/parsers.py @@ -8,6 +8,12 @@ __all__ = ["light_curve_parser"] +mission_sub_intervals = { + 'kepler': {'prefix': 'Q', 'card': 'QUARTER'}, + 'k2': {'prefix': 'C', 'card': 'CAMPAIGN'}, + 'tess': {'prefix': 'S', 'card': 'SECTOR'}, +} + @data_parser_registry("light_curve_parser") def light_curve_parser(app, file_obj, data_label=None, show_in_viewer=True, **kwargs): @@ -31,23 +37,25 @@ def light_curve_parser(app, file_obj, data_label=None, show_in_viewer=True, **kw else: raise NotImplementedError(f"could not parse light_curve with type {type(file_obj)}") - # make a data label: - if data_label is not None: - new_data_label = f'{data_label}' - else: - new_data_label = light_curve.meta.get('OBJECT', 'Light curve') - # handle flux_origin default mission = light_curve.meta.get('MISSION', '').lower() flux_origin = light_curve.meta.get('FLUX_ORIGIN', None) # i.e. PDCSAP or SAP - if isinstance(light_curve, lightkurve.targetpixelfile.TargetPixelFile): - new_data_label += '[TPF]' - elif mission == 'kepler': - new_data_label += f' Q{light_curve.meta.get("QUARTER")}' - elif mission == 'k2': - new_data_label += f' C{light_curve.meta.get("CAMPAIGN")}' - elif mission == 'tess': - new_data_label += f' S{light_curve.meta.get("SECTOR")}' + + # make a data label: + if data_label is None: + data_label = light_curve.meta.get('OBJECT', 'Light curve') + + if isinstance(light_curve, lightkurve.targetpixelfile.TargetPixelFile): + data_label += '[TPF]' + elif mission in mission_sub_intervals: + # the sub-interval label is something like "Q9" for Kepler or + # "S9" for TESS. If it's already in the proposed data label, skip; + # otherwise, append it. + sub_interval_label = ( + f' [{mission_sub_intervals[mission]["prefix"]}' + f'{light_curve.meta.get(mission_sub_intervals[mission]["card"])}]' + ) + data_label += sub_interval_label if flux_origin == 'flux' or (flux_origin is None and 'flux' in getattr(light_curve, 'columns', [])): # noqa # then make a copy of this column so it won't be lost when changing with the flux_column @@ -58,7 +66,7 @@ def light_curve_parser(app, file_obj, data_label=None, show_in_viewer=True, **kw light_curve.meta['FLUX_ORIGIN'] = 'flux:orig' data = _data_with_reftime(app, light_curve) - app.add_data(data, new_data_label) + app.add_data(data, data_label) if isinstance(light_curve, lightkurve.targetpixelfile.TargetPixelFile): # ensure an image/cube/TPF viewer exists @@ -69,12 +77,12 @@ def light_curve_parser(app, file_obj, data_label=None, show_in_viewer=True, **kw found_viewer = False for viewer_id, viewer in app._viewer_store.items(): if isinstance(viewer, CubeView): - app.add_data_to_viewer(viewer_id, new_data_label) + app.add_data_to_viewer(viewer_id, data_label) found_viewer = True if not found_viewer: app._on_new_viewer(NewViewerMessage(CubeView, data=None, sender=app), vid='image', name='image') - app.add_data_to_viewer('image', new_data_label) + app.add_data_to_viewer('image', data_label) # set TPF viewer's stretch to custom defaults: plot_options_plugin = PlotOptions(app=app) @@ -85,13 +93,13 @@ def light_curve_parser(app, file_obj, data_label=None, show_in_viewer=True, **kw if show_in_viewer: for viewer_id, viewer in app._viewer_store.items(): if isinstance(viewer, (TimeScatterView, PhaseScatterView)): - app.add_data_to_viewer(viewer_id, new_data_label) + app.add_data_to_viewer(viewer_id, data_label) # add to any known phase viewers ephem_plugin = app._jdaviz_helper.plugins.get('Ephemeris', None) if ephem_plugin is not None: for viewer in ephem_plugin._obj._get_phase_viewers(): - app.add_data_to_viewer(viewer.reference, new_data_label) + app.add_data_to_viewer(viewer.reference, data_label) def _data_with_reftime(app, light_curve): diff --git a/lcviz/tests/test_parser.py b/lcviz/tests/test_parser.py index a37b343e..ff3be3cc 100644 --- a/lcviz/tests/test_parser.py +++ b/lcviz/tests/test_parser.py @@ -119,3 +119,14 @@ def test_apply_yrangerois(helper, light_curve_like_kepler_quarter): subset_state = subsets['Subset 1'][0]['subset_state'] np.testing.assert_allclose([subset_state.lo, subset_state.hi], [1, 1.05]) + + +def test_data_label(helper, light_curve_like_kepler_quarter): + # add data without specifying data label: + helper.load_data(light_curve_like_kepler_quarter) + assert helper.app.data_collection[-1].label == 'Light curve [Q10]' + + # specify label, check that quarter isn't appended: + data_label = 'Cool target' + helper.load_data(light_curve_like_kepler_quarter, data_label=data_label) + assert helper.app.data_collection[-1].label == data_label diff --git a/lcviz/tests/test_plugin_markers.py b/lcviz/tests/test_plugin_markers.py index 85350cf8..b1853739 100644 --- a/lcviz/tests/test_plugin_markers.py +++ b/lcviz/tests/test_plugin_markers.py @@ -51,7 +51,7 @@ def test_plugin_markers(helper, light_curve_like_kepler_quarter): 'Time 5.45833e+00 d', 'Flux 9.67587e-01') - _assert_dict_allclose(label_mouseover.as_dict(), {'data_label': 'Light curve', + _assert_dict_allclose(label_mouseover.as_dict(), {'data_label': 'Light curve [Q10]', 'time': 5.4583335, 'time:unit': 'd', 'phase': np.nan, @@ -81,7 +81,7 @@ def test_plugin_markers(helper, light_curve_like_kepler_quarter): 'Phase 0.45833', 'Flux 9.67587e-01') - _assert_dict_allclose(label_mouseover.as_dict(), {'data_label': 'Light curve', + _assert_dict_allclose(label_mouseover.as_dict(), {'data_label': 'Light curve [Q10]', 'time': 5.458333374001086, 'time:unit': 'd', 'phase': 0.4583333730697632, diff --git a/pyproject.toml b/pyproject.toml index fd7c0707..1959839a 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -24,6 +24,7 @@ dependencies = [ # to devdeps in tox.ini "jdaviz>=3.10.2,<3.11", "lightkurve>=2.4.1", + "numpy<2", ] dynamic = [ "version",