diff --git a/CHANGES.rst b/CHANGES.rst index 76a5861034..fdf3397283 100644 --- a/CHANGES.rst +++ b/CHANGES.rst @@ -44,6 +44,7 @@ Bug Fixes Cubeviz ^^^^^^^ +- Removed the deprecated ``save as fits`` option from the Collapse, Moment Maps, and Spectral Extraction plugins; use the Export plugin instead. [#3256] Imviz ^^^^^ diff --git a/jdaviz/app.vue b/jdaviz/app.vue index 6ca646dc6f..5f67b89839 100644 --- a/jdaviz/app.vue +++ b/jdaviz/app.vue @@ -114,7 +114,7 @@
- {{ trayItem.label == 'Orientation' ? 'Orientation (prev. Links Control)' : trayItem.label }} + {{ trayItem.label }} diff --git a/jdaviz/components/tooltip.vue b/jdaviz/components/tooltip.vue index 341f97fa3b..a31a05a8ae 100644 --- a/jdaviz/components/tooltip.vue +++ b/jdaviz/components/tooltip.vue @@ -101,9 +101,7 @@ const tooltips = { 'plugin-line-lists-spectral-range': 'Toggle filter to only lines observable within the range of the Spectrum Viewer', 'plugin-line-analysis-sync-identify': 'Lock/unlock selection with identified line', 'plugin-line-analysis-assign': 'Assign the centroid wavelength and update the redshift', - 'plugin-moment-save-fits': 'Save moment map as FITS file', 'plugin-extract-save-fits': 'Save spectral extraction as FITS file', - 'plugin-collapse-save-fits': 'Save collapsed cube as FITS file', 'plugin-link-apply': 'Apply linking to data', 'plugin-footprints-color-picker': 'Change the color of the footprint overlay', 'plugin-dq-show-all': 'Show all quality flags', diff --git a/jdaviz/configs/cubeviz/plugins/moment_maps/moment_maps.py b/jdaviz/configs/cubeviz/plugins/moment_maps/moment_maps.py index ca4a099d1c..3aad940765 100644 --- a/jdaviz/configs/cubeviz/plugins/moment_maps/moment_maps.py +++ b/jdaviz/configs/cubeviz/plugins/moment_maps/moment_maps.py @@ -1,6 +1,3 @@ -import os -from pathlib import Path - import numpy as np import specutils from astropy import units as u @@ -80,7 +77,6 @@ class MomentMap(PluginTemplateMixin, DatasetSelectMixin, SpectralSubsetSelectMix n_moment = IntHandleEmpty(0).tag(sync=True) filename = Unicode().tag(sync=True) moment_available = Bool(False).tag(sync=True) - overwrite_warn = Bool(False).tag(sync=True) output_unit_items = List().tag(sync=True) output_radio_items = List().tag(sync=True) output_unit_selected = Unicode().tag(sync=True) @@ -360,45 +356,6 @@ def spectral_unit_selected(self): def vue_calculate_moment(self, *args): self.calculate_moment(add_data=True) - def vue_save_as_fits(self, *args): - self._write_moment_to_fits() - def vue_overwrite_fits(self, *args): """Attempt to force writing the moment map if the user confirms the desire to overwrite.""" self.overwrite_warn = False - self._write_moment_to_fits(overwrite=True) - - def _write_moment_to_fits(self, overwrite=False, *args): - if self.moment is None or not self.filename: # pragma: no cover - return - if not self.export_enabled: - # this should never be triggered since this is intended for UI-disabling and the - # UI section is hidden, but would prevent any JS-hacking - raise ValueError("Writing out moment map to file is currently disabled") - - # Make sure file does not end up in weird places in standalone mode. - path = os.path.dirname(self.filename) - if path and not os.path.exists(path): - raise ValueError(f"Invalid path={path}") - elif (not path or path.startswith("..")) and os.environ.get("JDAVIZ_START_DIR", ""): # noqa: E501 # pragma: no cover - filename = Path(os.environ["JDAVIZ_START_DIR"]) / self.filename - else: - filename = Path(self.filename).resolve() - - if filename.exists(): - if overwrite: - # Try to delete the file - filename.unlink() - if filename.exists(): - # Warn the user if the file still exists - raise FileExistsError(f"Unable to delete {filename}. Check user permissions.") - else: - self.overwrite_warn = True - return - - filename = str(filename) - self.moment.write(filename) - - # Let the user know where we saved the file. - self.hub.broadcast(SnackbarMessage( - f"Moment map saved to {os.path.abspath(filename)}", sender=self, color="success")) diff --git a/jdaviz/configs/cubeviz/plugins/moment_maps/moment_maps.vue b/jdaviz/configs/cubeviz/plugins/moment_maps/moment_maps.vue index 5cf4b7ea7b..3ffc5e73af 100644 --- a/jdaviz/configs/cubeviz/plugins/moment_maps/moment_maps.vue +++ b/jdaviz/configs/cubeviz/plugins/moment_maps/moment_maps.vue @@ -173,65 +173,5 @@ Cannot calculate moment: Must set reference wavelength for output in velocity units. - - Results - -
-
-
- - - - - - - - DeprecationWarning: Save as FITS functionality has moved to the Export plugin as of v3.9 and will be removed from Moment Maps plugin in a future release. - - - - - - Save as FITS - - - - -
-
- - - - - -
- A file with this name is already on disk. Overwrite? -
-
- - - - Cancel - Overwrite - - -
- -
- - -
diff --git a/jdaviz/configs/cubeviz/plugins/moment_maps/tests/test_moment_maps.py b/jdaviz/configs/cubeviz/plugins/moment_maps/tests/test_moment_maps.py index 4448193356..3aab8e711c 100644 --- a/jdaviz/configs/cubeviz/plugins/moment_maps/tests/test_moment_maps.py +++ b/jdaviz/configs/cubeviz/plugins/moment_maps/tests/test_moment_maps.py @@ -1,11 +1,9 @@ -import os import warnings from pathlib import Path import numpy as np import pytest from astropy import units as u -from astropy.io import fits from astropy.nddata import CCDData from astropy.tests.helper import assert_quantity_allclose from astropy.wcs import WCS @@ -144,11 +142,6 @@ def test_moment_calculation(cubeviz_helper, spectrum1d_cube, "World 13h39m59.9731s +27d00m00.3600s (ICRS)", "204.9998877673 27.0001000000 (deg)") - assert mm._obj.filename == 'moment0_test_FLUX.fits' # Auto-populated on calculate. - mm._obj.filename = str(tmp_path / mm._obj.filename) # But we want it in tmp_path for testing. - mm._obj.vue_save_as_fits() - assert os.path.isfile(mm._obj.filename) - mm._obj.n_moment = 1 mm._obj.output_unit_selected = "Spectral Unit" assert mm._obj.results_label == 'moment 1' @@ -258,7 +251,6 @@ def test_write_momentmap(cubeviz_helper, spectrum1d_cube, tmp_path): # Simulate an existing file on disk to check for overwrite warning test_file = tmp_path / "test_file.fits" - test_file_str = str(test_file) existing_sentinel_text = "This is a simulated, existing file on disk" test_file.write_text(existing_sentinel_text) @@ -266,34 +258,15 @@ def test_write_momentmap(cubeviz_helper, spectrum1d_cube, tmp_path): warnings.filterwarnings("ignore", message="No observer defined on WCS.*") cubeviz_helper.load_data(spectrum1d_cube, data_label='test') plugin = cubeviz_helper.plugins['Moment Maps'] - moment = plugin.calculate_moment() - - # By default, we shouldn't be warning the user of anything - assert plugin._obj.overwrite_warn is False + plugin.calculate_moment() - # Attempt to write the moment map to the same file we created earlier. - plugin._obj.filename = test_file_str - plugin._obj.vue_save_as_fits() - - # We should get an overwrite warning - assert plugin._obj.overwrite_warn is True # and shouldn't have written anything (the file should be intact) assert test_file.read_text() == existing_sentinel_text - # Force overwrite the existing file - plugin._obj.vue_overwrite_fits() - - # Read back in the file and check that it is equal to the one we calculated - with fits.open(test_file_str) as pf: - assert_allclose(pf[0].data, moment.data) - w = WCS(pf[0].header) - sky = w.pixel_to_world(0, 0) - assert_allclose(sky.ra.deg, 204.9998877673) - assert_allclose(sky.dec.deg, 27.0001) + label = plugin._obj.add_results.label + export_plugin = cubeviz_helper.plugins['Export']._obj - plugin._obj.filename = "fake_path/test_file.fits" - with pytest.raises(ValueError, match="Invalid path"): - plugin._obj.vue_save_as_fits() + assert label in export_plugin.data_collection.labels @pytest.mark.remote_data diff --git a/jdaviz/configs/cubeviz/plugins/spectral_extraction/spectral_extraction.py b/jdaviz/configs/cubeviz/plugins/spectral_extraction/spectral_extraction.py index f45d8db878..5d009cb76b 100644 --- a/jdaviz/configs/cubeviz/plugins/spectral_extraction/spectral_extraction.py +++ b/jdaviz/configs/cubeviz/plugins/spectral_extraction/spectral_extraction.py @@ -1,6 +1,4 @@ -import os from functools import cached_property -from pathlib import Path import numpy as np import astropy @@ -105,7 +103,6 @@ class SpectralExtraction(PluginTemplateMixin, ApertureSubsetSelectMixin, function_selected = Unicode('Sum').tag(sync=True) filename = Unicode().tag(sync=True) extraction_available = Bool(False).tag(sync=True) - overwrite_warn = Bool(False).tag(sync=True) results_units = Unicode().tag(sync=True) spectrum_y_units = Unicode().tag(sync=True) @@ -672,50 +669,6 @@ def vue_spectral_extraction(self, *args, **kwargs): def vue_create_bg_spec(self, *args, **kwargs): self.extract_bg_spectrum(add_data=True) - def vue_save_as_fits(self, *args): - self._save_extracted_spec_to_fits() - - def vue_overwrite_fits(self, *args): - """Attempt to force writing the spectral extraction if the user - confirms the desire to overwrite.""" - self.overwrite_warn = False - self._save_extracted_spec_to_fits(overwrite=True) - - def _save_extracted_spec_to_fits(self, overwrite=False, *args): - - if not self.export_enabled: - # this should never be triggered since this is intended for UI-disabling and the - # UI section is hidden, but would prevent any JS-hacking - raise ValueError(f"Writing out extracted {self.resulting_product_name} to file is currently disabled") # noqa - - # Make sure file does not end up in weird places in standalone mode. - path = os.path.dirname(self.filename) - if path and not os.path.exists(path): - raise ValueError(f"Invalid path={path}") - elif (not path or path.startswith("..")) and os.environ.get("JDAVIZ_START_DIR", ""): # noqa: E501 # pragma: no cover - filename = Path(os.environ["JDAVIZ_START_DIR"]) / self.filename - else: - filename = Path(self.filename).resolve() - - if filename.exists(): - if overwrite: - # Try to delete the file - filename.unlink() - if filename.exists(): - # Warn the user if the file still exists - raise FileExistsError(f"Unable to delete {filename}. Check user permissions.") - else: - self.overwrite_warn = True - return - - filename = str(filename) - self.extracted_spec.write(filename) - - # Let the user know where we saved the file. - self.hub.broadcast(SnackbarMessage( - f"Extracted {self.resulting_product_name} saved to {os.path.abspath(filename)}", - sender=self, color="success")) - @observe('aperture_selected', 'function_selected') def _set_default_results_label(self, event={}): if not hasattr(self, 'aperture'): diff --git a/jdaviz/configs/cubeviz/plugins/spectral_extraction/spectral_extraction.vue b/jdaviz/configs/cubeviz/plugins/spectral_extraction/spectral_extraction.vue index e30cb7d28d..2f0dbd12a6 100644 --- a/jdaviz/configs/cubeviz/plugins/spectral_extraction/spectral_extraction.vue +++ b/jdaviz/configs/cubeviz/plugins/spectral_extraction/spectral_extraction.vue @@ -271,65 +271,5 @@ - Results - -
-
-
- - - - - - - - - DeprecationWarning: Save as FITS functionality has moved to the Export plugin as of v3.9 and will be removed from here in a future release. - - - - - - Save as FITS - - - - -
-
- - - - - -
- A file with this name is already on disk. Overwrite? -
-
- - - - Cancel - Overwrite - - -
- -
-
-
- diff --git a/jdaviz/configs/cubeviz/plugins/spectral_extraction/tests/test_spectral_extraction.py b/jdaviz/configs/cubeviz/plugins/spectral_extraction/tests/test_spectral_extraction.py index 3d79e3cff1..6b0dc0cf57 100644 --- a/jdaviz/configs/cubeviz/plugins/spectral_extraction/tests/test_spectral_extraction.py +++ b/jdaviz/configs/cubeviz/plugins/spectral_extraction/tests/test_spectral_extraction.py @@ -144,7 +144,7 @@ def test_subset( assert_array_equal(collapsed_spec_2.uncertainty.array, expected_uncert) -def test_save_collapsed_to_fits(cubeviz_helper, spectrum1d_cube_with_uncerts, tmp_path): +def test_extracted_file_in_export_plugin(cubeviz_helper, spectrum1d_cube_with_uncerts, tmp_path): cubeviz_helper.load_data(spectrum1d_cube_with_uncerts) @@ -165,34 +165,10 @@ def test_save_collapsed_to_fits(cubeviz_helper, spectrum1d_cube_with_uncerts, tm assert extract_plugin._obj.filename == fname extract_plugin._obj.filename = str(fname_path) - # save output file with default name, make sure it exists - extract_plugin._obj.vue_save_as_fits() - assert fname_path.is_file() - - # read file back in, make sure it matches - dat = Spectrum1D.read(fname_path) - assert_array_equal(dat.data, extract_plugin._obj.extracted_spec.data) - assert dat.unit == extract_plugin._obj.extracted_spec.unit - - # make sure correct error message is raised when export_enabled is False - # this won't appear in UI, but just to be safe. - extract_plugin._obj.export_enabled = False - with pytest.raises( - ValueError, match="Writing out extracted spectrum to file is currently disabled"): - extract_plugin._obj.vue_save_as_fits() - extract_plugin._obj.export_enabled = True # set back to True - - # check that trying to overwrite without overwrite=True sets overwrite_warn to True, to - # display popup in UI - assert extract_plugin._obj.overwrite_warn is False - extract_plugin._obj.vue_save_as_fits() - assert extract_plugin._obj.overwrite_warn - - # check that writing out to a non existent directory fails as expected - extract_plugin._obj.filename = '/this/path/doesnt/exist.fits' - with pytest.raises(ValueError, match="Invalid path=/this/path/doesnt"): - extract_plugin._obj.vue_save_as_fits() - extract_plugin._obj.filename == fname # set back to original filename + label = extract_plugin._obj.add_results.label + export_plugin = cubeviz_helper.plugins['Export']._obj + + assert label in export_plugin.data_collection.labels def test_aperture_markers(cubeviz_helper, spectrum1d_cube): diff --git a/jdaviz/configs/default/plugins/collapse/collapse.py b/jdaviz/configs/default/plugins/collapse/collapse.py index 671051a51b..ad6382c0b1 100644 --- a/jdaviz/configs/default/plugins/collapse/collapse.py +++ b/jdaviz/configs/default/plugins/collapse/collapse.py @@ -1,5 +1,3 @@ -import os -from pathlib import Path import warnings from astropy.nddata import CCDData @@ -43,7 +41,6 @@ class Collapse(PluginTemplateMixin, DatasetSelectMixin, SpectralSubsetSelectMixi function_selected = Unicode('Sum').tag(sync=True) filename = Unicode().tag(sync=True) collapsed_spec_available = Bool(False).tag(sync=True) - overwrite_warn = Bool(False).tag(sync=True) # export_enabled controls whether saving to a file is enabled via the UI. This # is a temporary measure to allow server-installations to disable saving server-side until # saving client-side is supported @@ -143,46 +140,3 @@ def collapse(self, add_data=True): def vue_collapse(self, *args, **kwargs): self.collapse(add_data=True) - - def vue_save_as_fits(self, *args): - self._save_collapsed_spec_to_fits() - - def vue_overwrite_fits(self, *args): - """Attempt to force writing the file if the user confirms the desire to overwrite.""" - self.overwrite_warn = False - self._save_collapsed_spec_to_fits(overwrite=True) - - def _save_collapsed_spec_to_fits(self, overwrite=False, *args): - - if not self.export_enabled: - # this should never be triggered since this is intended for UI-disabling and the - # UI section is hidden, but would prevent any JS-hacking - raise ValueError("Writing out collapsed cube to file is currently disabled") - - # Make sure file does not end up in weird places in standalone mode. - path = os.path.dirname(self.filename) - if path and not os.path.exists(path): - raise ValueError(f"Invalid path={path}") - elif (not path or path.startswith("..")) and os.environ.get("JDAVIZ_START_DIR", ""): # noqa: E501 # pragma: no cover - filename = Path(os.environ["JDAVIZ_START_DIR"]) / self.filename - else: - filename = Path(self.filename).resolve() - - if filename.exists(): - if overwrite: - # Try to delete the file - filename.unlink() - if filename.exists(): - # Warn the user if the file still exists - raise FileExistsError(f"Unable to delete {filename}. Check user permissions.") - else: - self.overwrite_warn = True - return - - filename = str(filename) - self.collapsed_spec.write(filename) - - # Let the user know where we saved the file. - self.hub.broadcast(SnackbarMessage( - f"Collapsed cube saved to {os.path.abspath(filename)}", - sender=self, color="success")) diff --git a/jdaviz/configs/default/plugins/collapse/collapse.vue b/jdaviz/configs/default/plugins/collapse/collapse.vue index 196cf52862..8a2e4f9a49 100644 --- a/jdaviz/configs/default/plugins/collapse/collapse.vue +++ b/jdaviz/configs/default/plugins/collapse/collapse.vue @@ -59,65 +59,5 @@ :api_hints_enabled="api_hints_enabled" @click:action="collapse" > - - Results - -
-
-
- - - - - - - - - DeprecationWarning: Save as FITS functionality has moved to the Export plugin as of v3.9 and will be removed from here in a future release. - - - - - - Save as FITS - - - - -
-
- - - - - -
- A file with this name is already on disk. Overwrite? -
-
- - - - Cancel - Overwrite - - -
- -
- -
diff --git a/jdaviz/configs/default/plugins/collapse/tests/test_collapse.py b/jdaviz/configs/default/plugins/collapse/tests/test_collapse.py index d140afaeed..eec8067aa1 100644 --- a/jdaviz/configs/default/plugins/collapse/tests/test_collapse.py +++ b/jdaviz/configs/default/plugins/collapse/tests/test_collapse.py @@ -1,6 +1,5 @@ import numpy as np import pytest -from astropy.nddata import CCDData from astropy import units as u from specutils import Spectrum1D @@ -38,7 +37,7 @@ def test_linking_after_collapse(cubeviz_helper, spectral_cube_wcs): assert dc.external_links[3].cids2[0] == dc[-1].pixel_component_ids[0] -def test_save_collapsed_to_fits(cubeviz_helper, spectral_cube_wcs, tmp_path): +def test_collapsed_to_extract_plugin(cubeviz_helper, spectral_cube_wcs, tmp_path): cubeviz_helper.load_data(Spectrum1D(flux=np.ones((3, 4, 5)) * u.nJy, wcs=spectral_cube_wcs)) @@ -58,31 +57,7 @@ def test_save_collapsed_to_fits(cubeviz_helper, spectral_cube_wcs, tmp_path): assert collapse_plugin._obj.filename == fname collapse_plugin._obj.filename = str(tmp_path / fname) - # save output file with default name, make sure it exists - collapse_plugin._obj.vue_save_as_fits() - assert (tmp_path / fname).is_file() - - # read file back in, make sure it matches - dat = CCDData.read(collapse_plugin._obj.filename) - np.testing.assert_array_equal(dat.data, collapse_plugin._obj.collapsed_spec.data) - assert dat.unit == collapse_plugin._obj.collapsed_spec.unit - - # make sure correct error message is raised when export_enabled is False - # this won't appear in UI, but just to be safe. - collapse_plugin._obj.export_enabled = False - msg = "Writing out collapsed cube to file is currently disabled" - with pytest.raises(ValueError, match=msg): - collapse_plugin._obj.vue_save_as_fits() - collapse_plugin._obj.export_enabled = True # set back to True - - # check that trying to overwrite without overwrite=True sets overwrite_warn to True, to - # display popup in UI - assert collapse_plugin._obj.overwrite_warn is False - collapse_plugin._obj.vue_save_as_fits() - assert collapse_plugin._obj.overwrite_warn - - # check that writing out to a non existent directory fails as expected - collapse_plugin._obj.filename = '/this/path/doesnt/exist.fits' - with pytest.raises(ValueError, match="Invalid path=/this/path/doesnt"): - collapse_plugin._obj.vue_save_as_fits() - collapse_plugin._obj.filename == fname # set back to original filename + label = collapse_plugin._obj.add_results.label + export_plugin = cubeviz_helper.plugins['Export']._obj + + assert label in export_plugin.data_collection.labels