diff --git a/src/kbmod/file_utils.py b/src/kbmod/file_utils.py
index 86e59b453..b8297c0a6 100644
--- a/src/kbmod/file_utils.py
+++ b/src/kbmod/file_utils.py
@@ -75,10 +75,6 @@ class FileUtils:
Examples
--------
- * Load an external file of visit ID to timestamp mappings.
-
- ``time_dict = FileUtils.load_time_dictionary("kbmod/data/demo_times.dat")``
-
* Load the results of a KBMOD run as trajectory objects.
``FileUtils.load_results_file_as_trajectories("results_DEMO.txt")``
diff --git a/src/kbmod/search/layered_image.cpp b/src/kbmod/search/layered_image.cpp
index 50baaeda2..d07f18008 100644
--- a/src/kbmod/search/layered_image.cpp
+++ b/src/kbmod/search/layered_image.cpp
@@ -2,8 +2,9 @@
namespace search {
-LayeredImage::LayeredImage(const RawImage& sci, const RawImage& var, const RawImage& msk, const PSF& psf)
- : psf(psf) {
+LayeredImage::LayeredImage(const RawImage& sci, const RawImage& var, const RawImage& msk, const PSF& psf,
+ double obs_time)
+ : obstime(obs_time), psf(psf) {
// Get the dimensions of the science layer and check for consistency with
// the other two layers.
width = sci.get_width();
@@ -20,9 +21,10 @@ LayeredImage::LayeredImage(const RawImage& sci, const RawImage& var, const RawIm
}
LayeredImage::LayeredImage(Image& sci, Image& var, Image& msk, PSF& psf, double obs_time)
- : science(sci, obs_time), variance(var, obs_time), mask(msk, obs_time), psf(psf) {
+ : science(sci), variance(var), mask(msk), psf(psf) {
width = science.get_width();
height = science.get_height();
+ obstime = obs_time;
assert_sizes_equal(variance.get_width(), width, "variance layer width");
assert_sizes_equal(variance.get_height(), height, "variance layer height");
assert_sizes_equal(mask.get_width(), width, "mask layer width");
@@ -33,6 +35,7 @@ LayeredImage::LayeredImage(Image& sci, Image& var, Image& msk, PSF& psf, double
LayeredImage::LayeredImage(const LayeredImage& source) noexcept {
width = source.width;
height = source.height;
+ obstime = source.obstime;
science = source.science;
mask = source.mask;
variance = source.variance;
@@ -43,6 +46,7 @@ LayeredImage::LayeredImage(const LayeredImage& source) noexcept {
LayeredImage::LayeredImage(LayeredImage&& source) noexcept
: width(source.width),
height(source.height),
+ obstime(source.obstime),
science(std::move(source.science)),
mask(std::move(source.mask)),
variance(std::move(source.variance)),
@@ -52,6 +56,7 @@ LayeredImage::LayeredImage(LayeredImage&& source) noexcept
LayeredImage& LayeredImage::operator=(const LayeredImage& source) noexcept {
width = source.width;
height = source.height;
+ obstime = source.obstime;
science = source.science;
mask = source.mask;
variance = source.variance;
@@ -64,6 +69,7 @@ LayeredImage& LayeredImage::operator=(LayeredImage&& source) noexcept {
if (this != &source) {
width = source.width;
height = source.height;
+ obstime = source.obstime;
science = std::move(source.science);
mask = std::move(source.mask);
variance = std::move(source.variance);
@@ -252,7 +258,8 @@ static void layered_image_bindings(py::module& m) {
using pf = search::PSF;
py::class_
(m, "LayeredImage", pydocs::DOC_LayeredImage)
- .def(py::init())
+ .def(py::init(), py::arg("sci"),
+ py::arg("var"), py::arg("msk"), py::arg("psf"), py::arg("obs_time") = -1.0)
.def(py::init(),
py::arg("sci").noconvert(true), py::arg("var").noconvert(true),
py::arg("msk").noconvert(true), py::arg("psf"), py::arg("obs_time"))
diff --git a/src/kbmod/search/layered_image.h b/src/kbmod/search/layered_image.h
index d30127c86..292e31358 100644
--- a/src/kbmod/search/layered_image.h
+++ b/src/kbmod/search/layered_image.h
@@ -14,7 +14,8 @@
namespace search {
class LayeredImage {
public:
- explicit LayeredImage(const RawImage& sci, const RawImage& var, const RawImage& msk, const PSF& psf);
+ explicit LayeredImage(const RawImage& sci, const RawImage& var, const RawImage& msk, const PSF& psf,
+ double obs_time = -1.0);
// Build a layered image from the underlying matrices, taking ownership of the image data.
explicit LayeredImage(Image& sci, Image& var, Image& msk, PSF& psf, double obs_time);
@@ -33,8 +34,8 @@ class LayeredImage {
unsigned get_width() const { return width; }
unsigned get_height() const { return height; }
uint64_t get_npixels() const { return width * height; }
- double get_obstime() const { return science.get_obstime(); }
- void set_obstime(double obstime) { science.set_obstime(obstime); }
+ double get_obstime() const { return obstime; }
+ void set_obstime(double new_obstime) { obstime = new_obstime; }
// Getter functions for the data in the individual layers.
RawImage& get_science() { return science; }
@@ -92,6 +93,7 @@ class LayeredImage {
private:
unsigned width;
unsigned height;
+ double obstime;
PSF psf;
RawImage science;
diff --git a/src/kbmod/search/pydocs/layered_image_docs.h b/src/kbmod/search/pydocs/layered_image_docs.h
index 6691d49d8..70400bdc3 100644
--- a/src/kbmod/search/pydocs/layered_image_docs.h
+++ b/src/kbmod/search/pydocs/layered_image_docs.h
@@ -15,6 +15,8 @@ static const auto DOC_LayeredImage = R"doc(
The `RawImage` for the mask layer.
psf : `PSF`
The PSF for the image.
+ obstime : `float`
+ The time of the image.
Raises
------
diff --git a/src/kbmod/search/pydocs/raw_image_docs.h b/src/kbmod/search/pydocs/raw_image_docs.h
index 7a108a95e..94bae3386 100644
--- a/src/kbmod/search/pydocs/raw_image_docs.h
+++ b/src/kbmod/search/pydocs/raw_image_docs.h
@@ -5,23 +5,19 @@ namespace pydocs {
static const auto DOC_RawImage = R"doc(
Image and the time it was observed at.
- Instantiated from a pair of obstime and image, or from image dimensions and
- obstime. When instantiating from image dimensions and obstime, it's possible
- to provide a default value used to fill the array with. Otherwise the array is
- filled with zeros.
+ Instantiated from an image or from image dimensions and and a value (which
+ defaults to zero).
Parameters
----------
image : `numpy.array`, optional
Image, row-major a 2D array. The array *must* be of dtype `numpy.single`.
- obstime : `float`, optional
- MJD stamp, time the image was observed at.
width : `int`, optional
Width of the image.
height : `int`, optional
Height of the image.
value : `float`, optional
- When instantiated from dimensions and obstime, value that fills the array.
+ When instantiated from dimensions, value that fills the array.
Default is 0.
Attributes
@@ -32,8 +28,6 @@ static const auto DOC_RawImage = R"doc(
Image width, in pixels.
npixels : `int`
Number of pixels in the image, equivalent to ``width*height``.
- obstime : `float`
- MJD time of observation.
image : `np.array[np,single]`
Image array.
@@ -61,7 +55,7 @@ static const auto DOC_RawImage = R"doc(
>>> from kbmod.search import RawImage
>>> import numpy as np
>>> ri = RawImage()
- >>> ri = RawImage(w=2, h=3, value=1, obstime=10)
+ >>> ri = RawImage(w=2, h=3, value=1)
>>> ri.image
array([[1., 1.],
[1., 1.],
diff --git a/src/kbmod/search/raw_image.cpp b/src/kbmod/search/raw_image.cpp
index 6b1df1133..c27616cbf 100644
--- a/src/kbmod/search/raw_image.cpp
+++ b/src/kbmod/search/raw_image.cpp
@@ -4,17 +4,16 @@ namespace search {
using Index = indexing::Index;
using Point = indexing::Point;
-RawImage::RawImage() : width(0), height(0), obstime(-1.0), image() {}
+RawImage::RawImage() : width(0), height(0), image() {}
-RawImage::RawImage(Image& img, double obs_time) {
+RawImage::RawImage(Image& img) {
image = std::move(img);
height = image.rows();
width = image.cols();
- obstime = obs_time;
}
-RawImage::RawImage(unsigned w, unsigned h, float value, double obs_time)
- : width(w), height(h), obstime(obs_time) {
+RawImage::RawImage(unsigned w, unsigned h, float value)
+ : width(w), height(h) {
if (value != 0.0f)
image = Image::Constant(height, width, value);
else
@@ -26,14 +25,12 @@ RawImage::RawImage(const RawImage& old) noexcept {
width = old.get_width();
height = old.get_height();
image = old.get_image();
- obstime = old.get_obstime();
}
// Move constructor
RawImage::RawImage(RawImage&& source) noexcept
: width(source.width),
height(source.height),
- obstime(source.obstime),
image(std::move(source.image)) {}
// Copy assignment
@@ -41,7 +38,6 @@ RawImage& RawImage::operator=(const RawImage& source) noexcept {
width = source.width;
height = source.height;
image = source.image;
- obstime = source.obstime;
return *this;
}
@@ -51,7 +47,6 @@ RawImage& RawImage::operator=(RawImage&& source) noexcept {
width = source.width;
height = source.height;
image = std::move(source.image);
- obstime = source.obstime;
}
return *this;
}
@@ -484,15 +479,13 @@ static void raw_image_bindings(py::module& m) {
py::class_(m, "RawImage", pydocs::DOC_RawImage)
.def(py::init<>())
.def(py::init())
- .def(py::init(), py::arg("img").noconvert(true),
- py::arg("obs_time") = -1.0d)
- .def(py::init(), py::arg("w"), py::arg("h"),
- py::arg("value") = 0.0f, py::arg("obs_time") = -1.0d)
+ .def(py::init(), py::arg("img").noconvert(true))
+ .def(py::init(), py::arg("w"), py::arg("h"),
+ py::arg("value") = 0.0f)
// attributes and properties
.def_property_readonly("height", &rie::get_height)
.def_property_readonly("width", &rie::get_width)
.def_property_readonly("npixels", &rie::get_npixels)
- .def_property("obstime", &rie::get_obstime, &rie::set_obstime)
.def_property("image", py::overload_cast<>(&rie::get_image, py::const_), &rie::set_image)
.def_property("imref", py::overload_cast<>(&rie::get_image), &rie::set_image)
// pixel accessors and setters
diff --git a/src/kbmod/search/raw_image.h b/src/kbmod/search/raw_image.h
index 42c49bd35..c135761c1 100644
--- a/src/kbmod/search/raw_image.h
+++ b/src/kbmod/search/raw_image.h
@@ -26,8 +26,8 @@ using ImageIRef = Eigen::Ref;
class RawImage {
public:
explicit RawImage();
- explicit RawImage(Image& img, double obs_time = -1.0);
- explicit RawImage(unsigned w, unsigned h, float value = 0.0, double obs_time = -1.0);
+ explicit RawImage(Image& img);
+ explicit RawImage(unsigned w, unsigned h, float value = 0.0);
RawImage(const RawImage& old) noexcept; // Copy constructor
RawImage(RawImage&& source) noexcept; // Move constructor
@@ -70,10 +70,6 @@ class RawImage {
void replace_masked_values(float value = 0.0);
- // Functions for locally storing the image time.
- double get_obstime() const { return obstime; }
- void set_obstime(double new_time) { obstime = new_time; }
-
// this will be a raw pointer to the underlying array
// we use this to copy to GPU and nowhere else!
float* data() { return image.data(); }
@@ -129,7 +125,6 @@ class RawImage {
private:
unsigned width;
unsigned height;
- double obstime;
Image image;
};
diff --git a/src/kbmod/standardizers/fits_standardizers/fits_standardizer.py b/src/kbmod/standardizers/fits_standardizers/fits_standardizer.py
index dfce61adc..28efbe11d 100644
--- a/src/kbmod/standardizers/fits_standardizers/fits_standardizer.py
+++ b/src/kbmod/standardizers/fits_standardizers/fits_standardizer.py
@@ -412,5 +412,5 @@ def toLayeredImage(self):
for sci, var, mask, psf, t in zip(sciences, variances, masks, psfs, mjds):
# Converts nd array mask from bool to np.float32
mask = mask.astype(np.float32)
- imgs.append(LayeredImage(RawImage(sci, t), RawImage(var, t), RawImage(mask, t), psf))
+ imgs.append(LayeredImage(RawImage(sci), RawImage(var), RawImage(mask), psf, obs_time=t))
return imgs
diff --git a/src/kbmod/work_unit.py b/src/kbmod/work_unit.py
index bf9e932bb..0603e3197 100644
--- a/src/kbmod/work_unit.py
+++ b/src/kbmod/work_unit.py
@@ -389,22 +389,23 @@ def to_fits(self, filename, overwrite=False):
for i in range(self.im_stack.img_count()):
layered = self.im_stack.get_single_image(i)
+ obstime = layered.get_obstime()
c_indices = self._per_image_indices[i]
n_indices = len(c_indices)
img_wcs = self.get_wcs(i)
- sci_hdu = raw_image_to_hdu(layered.get_science(), img_wcs)
+ sci_hdu = raw_image_to_hdu(layered.get_science(), obstime, img_wcs)
sci_hdu.name = f"SCI_{i}"
sci_hdu.header["NIND"] = n_indices
for j in range(n_indices):
sci_hdu.header[f"IND_{j}"] = c_indices[j]
hdul.append(sci_hdu)
- var_hdu = raw_image_to_hdu(layered.get_variance())
+ var_hdu = raw_image_to_hdu(layered.get_variance(), obstime)
var_hdu.name = f"VAR_{i}"
hdul.append(var_hdu)
- msk_hdu = raw_image_to_hdu(layered.get_mask())
+ msk_hdu = raw_image_to_hdu(layered.get_mask(), obstime)
msk_hdu.name = f"MSK_{i}"
hdul.append(msk_hdu)
@@ -463,23 +464,24 @@ def to_sharded_fits(self, filename, directory, overwrite=False):
for i in range(self.im_stack.img_count()):
layered = self.im_stack.get_single_image(i)
+ obstime = layered.get_obstime()
c_indices = self._per_image_indices[i]
n_indices = len(c_indices)
sub_hdul = fits.HDUList()
img_wcs = self.get_wcs(i)
- sci_hdu = raw_image_to_hdu(layered.get_science(), img_wcs)
+ sci_hdu = raw_image_to_hdu(layered.get_science(), obstime, img_wcs)
sci_hdu.name = f"SCI_{i}"
sci_hdu.header["NIND"] = n_indices
for j in range(n_indices):
sci_hdu.header[f"IND_{j}"] = c_indices[j]
sub_hdul.append(sci_hdu)
- var_hdu = raw_image_to_hdu(layered.get_variance())
+ var_hdu = raw_image_to_hdu(layered.get_variance(), obstime)
var_hdu.name = f"VAR_{i}"
sub_hdul.append(var_hdu)
- msk_hdu = raw_image_to_hdu(layered.get_mask())
+ msk_hdu = raw_image_to_hdu(layered.get_mask(), obstime)
msk_hdu.name = f"MSK_{i}"
sub_hdul.append(msk_hdu)
@@ -835,13 +837,15 @@ def load_layered_image_from_shard(file_path):
return img
-def raw_image_to_hdu(img, wcs=None):
+def raw_image_to_hdu(img, obstime, wcs=None):
"""Helper function that creates a HDU out of RawImage.
Parameters
----------
img : `RawImage`
The RawImage to convert.
+ obstime : `float`
+ The time of the observation.
wcs : `astropy.wcs.WCS`
An optional WCS to include in the header.
@@ -857,28 +861,6 @@ def raw_image_to_hdu(img, wcs=None):
append_wcs_to_hdu_header(wcs, hdu.header)
# Set the time stamp.
- hdu.header["MJD"] = img.obstime
- return hdu
-
-
-def hdu_to_raw_image(hdu):
- """Helper function that creates a RawImage from a HDU.
-
- Parameters
- ----------
- hdu : `astropy.io.fits.hdu.image.ImageHDU`
- The image extension.
+ hdu.header["MJD"] = obstime
- Returns
- -------
- img : `RawImage` or None
- The RawImage if there is valid data and None otherwise.
- """
- img = None
- if isinstance(hdu, fits.hdu.image.ImageHDU):
- # This will be a copy whenever dtype != np.single including when
- # endianness doesn't match the native float.
- img = RawImage(hdu.data.astype(np.single))
- if "MJD" in hdu.header:
- img.obstime = hdu.header["MJD"]
- return img
+ return hdu
diff --git a/tests/test_butlerstd.py b/tests/test_butlerstd.py
index 8647ada79..ba810af09 100644
--- a/tests/test_butlerstd.py
+++ b/tests/test_butlerstd.py
@@ -223,9 +223,6 @@ def test_to_layered_image(self):
# times can only be compred approximately, because sometimes we
# calculate the time in the middle of the exposure
self.assertAlmostEqual(expected_mjd, img.get_obstime(), 2)
- self.assertAlmostEqual(expected_mjd, img.get_science().obstime, 2)
- self.assertAlmostEqual(expected_mjd, img.get_variance().obstime, 2)
- self.assertAlmostEqual(expected_mjd, img.get_mask().obstime, 2)
if __name__ == "__main__":
diff --git a/tests/test_fake_data_creator.py b/tests/test_fake_data_creator.py
index ea92205b8..8bc923623 100644
--- a/tests/test_fake_data_creator.py
+++ b/tests/test_fake_data_creator.py
@@ -25,7 +25,7 @@ def test_create_fake_times(self):
self.assertAlmostEqual(times2[i], expected[i])
def test_add_fake_object(self):
- img = RawImage(20, 10, 0.0, 1.0) # All zero image.
+ img = RawImage(20, 10, 0.0) # All zero image.
p = PSF(np.full((3, 3), 1.0 / 9.0)) # Equal PSF.
add_fake_object(img, 5.5, 3.5, 100.0, p)
diff --git a/tests/test_psi_phi_array.py b/tests/test_psi_phi_array.py
index 900ed6384..050db096c 100644
--- a/tests/test_psi_phi_array.py
+++ b/tests/test_psi_phi_array.py
@@ -30,14 +30,14 @@ def setUp(self):
psi_1_vals = np.arange(0, self.width * self.height, dtype=np.single)
psi_1_arr = psi_1_vals.reshape(self.height, self.width)
- self.psi_1 = RawImage(img=psi_1_arr, obs_time=1.0)
+ self.psi_1 = RawImage(img=psi_1_arr)
psi_2_vals = np.arange(self.width * self.height, 2 * self.width * self.height, dtype=np.single)
psi_2_arr = psi_2_vals.reshape(self.height, self.width)
- self.psi_2 = RawImage(img=psi_2_arr, obs_time=2.0)
+ self.psi_2 = RawImage(img=psi_2_arr)
- self.phi_1 = RawImage(np.full((self.height, self.width), 0.1, dtype=np.single), obs_time=1.0)
- self.phi_2 = RawImage(np.full((self.height, self.width), 0.2, dtype=np.single), obs_time=2.0)
+ self.phi_1 = RawImage(np.full((self.height, self.width), 0.1, dtype=np.single))
+ self.phi_2 = RawImage(np.full((self.height, self.width), 0.2, dtype=np.single))
self.zeroed_times = [0.0, 1.0]
diff --git a/tests/test_raw_image.py b/tests/test_raw_image.py
index fa4ddadc2..9bd86a3c7 100644
--- a/tests/test_raw_image.py
+++ b/tests/test_raw_image.py
@@ -37,55 +37,47 @@ def test_create(self):
img = RawImage()
self.assertEqual(img.width, 0)
self.assertEqual(img.height, 0)
- self.assertEqual(img.obstime, -1.0)
# from NumPy arrays
- img = RawImage(img=self.array, obs_time=10.0)
+ img = RawImage(img=self.array)
self.assertEqual(img.image.shape, (self.height, self.width))
- self.assertEqual(img.obstime, 10.0)
self.assertEqual(img.npixels, self.width * self.height)
self.assertTrue((img.image == self.array).all())
img2 = RawImage(img=self.array)
self.assertTrue((img2.image == img.image).all())
- self.assertEqual(img2.obstime, -1.0)
# from dimensions
img = RawImage(self.width, self.height)
self.assertEqual(img.image.shape, (self.height, self.width))
- self.assertEqual(img.obstime, -1.0)
self.assertTrue((img.image == 0).all())
# dimensions and optional values
img = RawImage(self.height, self.width, 10)
self.assertTrue((img.image == 10).all())
- img = RawImage(self.height, self.width, 10, 12.5)
+ img = RawImage(self.height, self.width, 10)
self.assertTrue((img.image == 10).all())
- self.assertEqual(img.obstime, 12.5)
- img = RawImage(self.height, self.width, value=7.5, obs_time=12.5)
+ img = RawImage(self.height, self.width, value=7.5)
self.assertTrue((img.image == 7.5).all())
- self.assertEqual(img.obstime, 12.5)
# copy constructor, set the old image to all zeros and change the time.
- img = RawImage(img=self.array, obs_time=10.0)
+ img = RawImage(img=self.array)
img2 = RawImage(img)
img.set_all(0.0)
- img.obstime = 1.0
self.assertTrue((img2.image == self.array).all())
- self.assertEqual(img2.obstime, 10.0)
def test_pixel_getters(self):
"""Test RawImage masked pixel value getters"""
- img = RawImage(img=self.array, obs_time=10.0)
+ img = RawImage(img=self.array)
self.assertFalse(pixel_value_valid(img.get_pixel(-1, 5)))
self.assertFalse(pixel_value_valid(img.get_pixel(5, self.width)))
self.assertFalse(pixel_value_valid(img.get_pixel(5, -1)))
self.assertFalse(pixel_value_valid(img.get_pixel(self.height, 5)))
def test_contains(self):
- img = RawImage(img=self.array, obs_time=10.0)
+ img = RawImage(img=self.array)
self.assertTrue(img.contains_index(0, 0))
self.assertTrue(img.contains_index(1, 2))
self.assertFalse(img.contains_index(1, -1))
@@ -102,7 +94,7 @@ def test_contains(self):
self.assertFalse(img.contains_point(1.0, self.height + 1e-4))
def test_validity_checker(self):
- img = RawImage(img=np.array([[0, 0], [0, 0]]).astype(np.float32), obs_time=10.0)
+ img = RawImage(img=np.array([[0, 0], [0, 0]]).astype(np.float32))
self.assertTrue(img.pixel_has_data(0, 0))
img.set_pixel(0, 0, np.nan)
@@ -119,7 +111,7 @@ def test_validity_checker(self):
def test_interpolated_add(self):
"""Test that we can add values to the pixel."""
- img = RawImage(img=self.array, obs_time=10.0)
+ img = RawImage(img=self.array)
# Get the original value using (r, c) lookup.
org_val17 = img.get_pixel(1, 7)
@@ -153,7 +145,7 @@ def test_interpolated_add(self):
def test_approx_equal(self):
"""Test RawImage pixel value setters."""
- img = RawImage(img=self.array, obs_time=10.0)
+ img = RawImage(img=self.array)
# This test is testing L^\infy norm closeness. Eigen isApprox uses L2
# norm closeness.
diff --git a/tests/test_standardizer.py b/tests/test_standardizer.py
index 750224f52..cfd0d9a69 100644
--- a/tests/test_standardizer.py
+++ b/tests/test_standardizer.py
@@ -364,9 +364,6 @@ def test_to_layered_image(self):
# Test that we correctly set metadata
self.assertEqual(expected_mjd, img.get_obstime())
- self.assertEqual(expected_mjd, img.get_science().obstime)
- self.assertEqual(expected_mjd, img.get_variance().obstime)
- self.assertEqual(expected_mjd, img.get_mask().obstime)
if __name__ == "__main__":
diff --git a/tests/test_work_unit.py b/tests/test_work_unit.py
index dbcf7b8ed..01cb30224 100644
--- a/tests/test_work_unit.py
+++ b/tests/test_work_unit.py
@@ -15,7 +15,7 @@
import kbmod.search as kb
from kbmod.reprojection_utils import fit_barycentric_wcs
from kbmod.wcs_utils import make_fake_wcs, wcs_fits_equal
-from kbmod.work_unit import hdu_to_raw_image, raw_image_to_hdu, WorkUnit
+from kbmod.work_unit import raw_image_to_hdu, WorkUnit
import numpy.testing as npt