diff --git a/tests/test_metadata.py b/tests/test_metadata.py index 5a482673f..28ac81dc7 100644 --- a/tests/test_metadata.py +++ b/tests/test_metadata.py @@ -8,6 +8,7 @@ Band, BandDimension, CollectionMetadata, + CubeMetadata, Dimension, DimensionAlreadyExistsException, MetadataException, @@ -17,12 +18,7 @@ def test_metadata_get(): - metadata = CollectionMetadata({ - "foo": "bar", - "very": { - "deeply": {"nested": {"path": {"to": "somewhere"}}} - } - }) + metadata = CollectionMetadata({"foo": "bar", "very": {"deeply": {"nested": {"path": {"to": "somewhere"}}}}}) assert metadata.get("foo") == "bar" assert metadata.get("very", "deeply", "nested", "path", "to") == "somewhere" assert metadata.get("invalid", "key") is None @@ -30,9 +26,7 @@ def test_metadata_get(): def test_metadata_extent(): - metadata = CollectionMetadata({ - "extent": {"spatial": {"xmin": 4, "xmax": 10}} - }) + metadata = CollectionMetadata({"extent": {"spatial": {"xmin": 4, "xmax": 10}}}) assert metadata.extent == {"spatial": {"xmin": 4, "xmax": 10}} @@ -42,22 +36,28 @@ def test_band_minimal(): def test_band_dimension(): - bdim = BandDimension(name="spectral", bands=[ - Band("B02", "blue", 0.490), - Band("B03", "green", 0.560), - Band("B04", "red", 0.665), - ]) + bdim = BandDimension( + name="spectral", + bands=[ + Band("B02", "blue", 0.490), + Band("B03", "green", 0.560), + Band("B04", "red", 0.665), + ], + ) assert bdim.band_names == ["B02", "B03", "B04"] assert bdim.common_names == ["blue", "green", "red"] def test_band_dimension_band_index(): - bdim = BandDimension(name="spectral", bands=[ - Band("B02", "blue", 0.490), - Band("B03", "green", 0.560), - Band("B04", "red", 0.665), - Band("B08", "nir", 0.842), - ]) + bdim = BandDimension( + name="spectral", + bands=[ + Band("B02", "blue", 0.490), + Band("B03", "green", 0.560), + Band("B04", "red", 0.665), + Band("B08", "nir", 0.842), + ], + ) assert bdim.band_index(0) == 0 assert bdim.band_index(2) == 2 with pytest.raises(ValueError, match="Invalid band name/index"): @@ -78,10 +78,13 @@ def test_band_dimension_band_index(): def test_band_dimension_band_name(): - bdim = BandDimension(name="spectral", bands=[ - Band("B02", "blue", 0.490), - Band("B03", "green", 0.560), - ]) + bdim = BandDimension( + name="spectral", + bands=[ + Band("B02", "blue", 0.490), + Band("B03", "green", 0.560), + ], + ) assert bdim.band_name("B02") == "B02" assert bdim.band_name("B03") == "B03" with pytest.raises(ValueError, match="Invalid band name/index"): @@ -111,20 +114,30 @@ def test_band_dimension_rename_labels(): b03 = Band("B03", "green", 0.560) b04 = Band("B04", "red", 0.665) bdim = BandDimension(name="bs", bands=[b02, b03, b04]) + metadata = CollectionMetadata({}, dimensions=[bdim]) - newdim = metadata.rename_labels("bs", target=['1', '2', '3']).band_dimension + newdim = metadata.rename_labels("bs", target=["1", "2", "3"]).band_dimension + assert metadata.band_dimension.band_names == ["B02", "B03", "B04"] + assert newdim.band_names == ["1", "2", "3"] - assert metadata.band_dimension.band_names == ['B02', 'B03', 'B04'] - assert newdim.band_names == ['1', '2', '3'] + metadata = CubeMetadata(dimensions=[bdim]) + newdim = metadata.rename_labels("bs", target=["1", "2", "3"]).band_dimension + assert metadata.band_dimension.band_names == ["B02", "B03", "B04"] + assert newdim.band_names == ["1", "2", "3"] def test_band_dimension_set_labels(): - bdim = BandDimension(name="bs", bands=[Band('some_name', None, None)]) + bdim = BandDimension(name="bs", bands=[Band("some_name", None, None)]) + metadata = CollectionMetadata({}, dimensions=[bdim]) - newdim = metadata.rename_labels("bs", target=['1', '2', '3']).band_dimension + newdim = metadata.rename_labels("bs", target=["1", "2", "3"]).band_dimension + assert metadata.band_dimension.band_names == ["some_name"] + assert newdim.band_names == ["1", "2", "3"] - assert metadata.band_dimension.band_names == ['some_name'] - assert newdim.band_names == ['1', '2', '3'] + metadata = CubeMetadata(dimensions=[bdim]) + newdim = metadata.rename_labels("bs", target=["1", "2", "3"]).band_dimension + assert metadata.band_dimension.band_names == ["some_name"] + assert newdim.band_names == ["1", "2", "3"] def test_band_dimension_rename_labels_with_source(): @@ -132,20 +145,30 @@ def test_band_dimension_rename_labels_with_source(): b03 = Band("B03", "green", 0.560) b04 = Band("B04", "red", 0.665) bdim = BandDimension(name="bs", bands=[b02, b03, b04]) + metadata = CollectionMetadata({}, dimensions=[bdim]) - newdim = metadata.rename_labels("bs", target=['2'], source=['B03']).band_dimension + newdim = metadata.rename_labels("bs", target=["2"], source=["B03"]).band_dimension + assert metadata.band_dimension.band_names == ["B02", "B03", "B04"] + assert newdim.band_names == ["B02", "2", "B04"] - assert metadata.band_dimension.band_names == ['B02', 'B03', 'B04'] - assert newdim.band_names == ['B02', '2', 'B04'] + metadata = CubeMetadata(dimensions=[bdim]) + newdim = metadata.rename_labels("bs", target=["2"], source=["B03"]).band_dimension + assert metadata.band_dimension.band_names == ["B02", "B03", "B04"] + assert newdim.band_names == ["B02", "2", "B04"] def test_band_dimension_rename_labels_with_source_mismatch(): b02 = Band("B02", "blue", 0.490) b03 = Band("B03", "green", 0.560) bdim = BandDimension(name="bs", bands=[b02, b03]) + metadata = CollectionMetadata({}, dimensions=[bdim]) with pytest.raises(ValueError, match="should have same number of labels, but got"): - _ = metadata.rename_labels("bs", target=['2', "3"], source=['B03']) + _ = metadata.rename_labels("bs", target=["2", "3"], source=["B03"]) + + metadata = CubeMetadata(dimensions=[bdim]) + with pytest.raises(ValueError, match="should have same number of labels, but got"): + _ = metadata.rename_labels("bs", target=["2", "3"], source=["B03"]) def assert_same_dimensions(dims1: List[Dimension], dims2: List[Dimension]): @@ -158,219 +181,309 @@ def test_get_dimensions_cube_dimensions_empty(): def test_get_dimensions_cube_dimensions_spatial_xyt(): - dims = CollectionMetadata._parse_dimensions({ - "cube:dimensions": { - "xx": {"type": "spatial", "extent": [-10, 10]}, - "yy": {"type": "spatial", "extent": [-56, 83], "reference_system": 123}, - "tt": {"type": "temporal", "extent": ["2020-02-20", None]}, + dims = CollectionMetadata._parse_dimensions( + { + "cube:dimensions": { + "xx": {"type": "spatial", "extent": [-10, 10]}, + "yy": {"type": "spatial", "extent": [-56, 83], "reference_system": 123}, + "tt": {"type": "temporal", "extent": ["2020-02-20", None]}, + } } - }) - assert_same_dimensions(dims, [ - SpatialDimension(name="xx", extent=[-10, 10]), - SpatialDimension(name="yy", extent=[-56, 83], crs=123), - TemporalDimension(name="tt", extent=["2020-02-20", None]), - ]) + ) + assert_same_dimensions( + dims, + [ + SpatialDimension(name="xx", extent=[-10, 10]), + SpatialDimension(name="yy", extent=[-56, 83], crs=123), + TemporalDimension(name="tt", extent=["2020-02-20", None]), + ], + ) def test_get_dimensions_cube_dimensions_spatial_xyt_bands(): - dims = CollectionMetadata._parse_dimensions({ - "cube:dimensions": { - "x": {"type": "spatial", "extent": [-10, 10]}, - "y": {"type": "spatial", "extent": [-56, 83], "reference_system": 123}, - "t": {"type": "temporal", "extent": ["2020-02-20", None]}, - "spectral": {"type": "bands", "values": ["red", "green", "blue"]}, + dims = CollectionMetadata._parse_dimensions( + { + "cube:dimensions": { + "x": {"type": "spatial", "extent": [-10, 10]}, + "y": {"type": "spatial", "extent": [-56, 83], "reference_system": 123}, + "t": {"type": "temporal", "extent": ["2020-02-20", None]}, + "spectral": {"type": "bands", "values": ["red", "green", "blue"]}, + } } - }) - assert_same_dimensions(dims, [ - SpatialDimension(name="x", extent=[-10, 10]), - SpatialDimension(name="y", extent=[-56, 83], crs=123), - TemporalDimension(name="t", extent=["2020-02-20", None]), - BandDimension(name="spectral", bands=[ - Band("red", None, None), - Band("green", None, None), - Band("blue", None, None), - ]) - ]) + ) + assert_same_dimensions( + dims, + [ + SpatialDimension(name="x", extent=[-10, 10]), + SpatialDimension(name="y", extent=[-56, 83], crs=123), + TemporalDimension(name="t", extent=["2020-02-20", None]), + BandDimension( + name="spectral", + bands=[ + Band("red", None, None), + Band("green", None, None), + Band("blue", None, None), + ], + ), + ], + ) def test_get_dimensions_cube_dimensions_non_standard_type(): logs = [] - dims = CollectionMetadata._parse_dimensions({ - "cube:dimensions": { - "bar": {"type": "foo"}, + dims = CollectionMetadata._parse_dimensions( + { + "cube:dimensions": { + "bar": {"type": "foo"}, + }, }, - }, complain=logs.append) - assert_same_dimensions(dims, [ - Dimension(type="foo", name="bar") - ]) + complain=logs.append, + ) + assert_same_dimensions(dims, [Dimension(type="foo", name="bar")]) assert logs == ["Unknown dimension type 'foo'"] def test_get_dimensions_cube_dimensions_no_band_names(): logs = [] - dims = CollectionMetadata._parse_dimensions({ - "cube:dimensions": { - "spectral": {"type": "bands"}, + dims = CollectionMetadata._parse_dimensions( + { + "cube:dimensions": { + "spectral": {"type": "bands"}, + }, }, - }, complain=logs.append) - assert_same_dimensions(dims, [ - BandDimension(name="spectral", bands=[]) - ]) + complain=logs.append, + ) + assert_same_dimensions(dims, [BandDimension(name="spectral", bands=[])]) assert logs == ["No band names in dimension 'spectral'"] def test_get_dimensions_cube_dimensions_eo_bands(): - dims = CollectionMetadata._parse_dimensions({ - "cube:dimensions": { - "x": {"type": "spatial", "extent": [-10, 10]}, - "y": {"type": "spatial", "extent": [-56, 83], "reference_system": 123}, - "t": {"type": "temporal", "extent": ["2020-02-20", None]}, - "spectral": {"type": "bands", "values": ["r", "g", "b"]}, - }, - "summaries": { - "eo:bands": [ - {"name": "r", "common_name": "red", "center_wavelength": 5}, - {"name": "g", "center_wavelength": 8}, - {"name": "b", "common_name": "blue"}, - ] + dims = CollectionMetadata._parse_dimensions( + { + "cube:dimensions": { + "x": {"type": "spatial", "extent": [-10, 10]}, + "y": {"type": "spatial", "extent": [-56, 83], "reference_system": 123}, + "t": {"type": "temporal", "extent": ["2020-02-20", None]}, + "spectral": {"type": "bands", "values": ["r", "g", "b"]}, + }, + "summaries": { + "eo:bands": [ + {"name": "r", "common_name": "red", "center_wavelength": 5}, + {"name": "g", "center_wavelength": 8}, + {"name": "b", "common_name": "blue"}, + ] + }, } - }) - assert_same_dimensions(dims, [ - SpatialDimension(name="x", extent=[-10, 10]), - SpatialDimension(name="y", extent=[-56, 83], crs=123), - TemporalDimension(name="t", extent=["2020-02-20", None]), - BandDimension(name="spectral", bands=[ - Band("r", "red", 5), - Band("g", None, 8), - Band("b", "blue", None), - ]) - ]) + ) + assert_same_dimensions( + dims, + [ + SpatialDimension(name="x", extent=[-10, 10]), + SpatialDimension(name="y", extent=[-56, 83], crs=123), + TemporalDimension(name="t", extent=["2020-02-20", None]), + BandDimension( + name="spectral", + bands=[ + Band("r", "red", 5), + Band("g", None, 8), + Band("b", "blue", None), + ], + ), + ], + ) def test_get_dimensions_cube_dimensions_eo_bands_mismatch(): logs = [] - dims = CollectionMetadata._parse_dimensions({ - "cube:dimensions": { - "x": {"type": "spatial", "extent": [-10, 10]}, - "spectral": {"type": "bands", "values": ["r", "g", "b"]}, + dims = CollectionMetadata._parse_dimensions( + { + "cube:dimensions": { + "x": {"type": "spatial", "extent": [-10, 10]}, + "spectral": {"type": "bands", "values": ["r", "g", "b"]}, + }, + "summaries": { + "eo:bands": [ + {"name": "y", "common_name": "yellow", "center_wavelength": 5}, + {"name": "c", "center_wavelength": 8}, + {"name": "m", "common_name": "magenta"}, + ] + }, }, - "summaries": { - "eo:bands": [ - {"name": "y", "common_name": "yellow", "center_wavelength": 5}, - {"name": "c", "center_wavelength": 8}, - {"name": "m", "common_name": "magenta"}, - ] - } - }, complain=logs.append) - assert_same_dimensions(dims, [ - SpatialDimension(name="x", extent=[-10, 10]), - BandDimension(name="spectral", bands=[ - Band("r", None, None), - Band("g", None, None), - Band("b", None, None), - ]) - ]) + complain=logs.append, + ) + assert_same_dimensions( + dims, + [ + SpatialDimension(name="x", extent=[-10, 10]), + BandDimension( + name="spectral", + bands=[ + Band("r", None, None), + Band("g", None, None), + Band("b", None, None), + ], + ), + ], + ) assert logs == ["Band name mismatch: ['r', 'g', 'b'] != ['y', 'c', 'm']"] def test_get_dimensions_eo_bands_only(): logs = [] - dims = CollectionMetadata._parse_dimensions({ - "summaries": { - "eo:bands": [ - {"name": "y", "common_name": "yellow", "center_wavelength": 5}, - {"name": "c", "center_wavelength": 8}, - {"name": "m", "common_name": "magenta"}, - ] - } - }, complain=logs.append) - assert_same_dimensions(dims, [ - BandDimension(name="bands", bands=[ - Band("y", "yellow", 5), - Band("c", None, 8), - Band("m", "magenta", None), - ]) - ]) - assert logs == [ - 'No cube:dimensions metadata', - "Assuming name 'bands' for anonymous band dimension." - ] + dims = CollectionMetadata._parse_dimensions( + { + "summaries": { + "eo:bands": [ + {"name": "y", "common_name": "yellow", "center_wavelength": 5}, + {"name": "c", "center_wavelength": 8}, + {"name": "m", "common_name": "magenta"}, + ] + } + }, + complain=logs.append, + ) + assert_same_dimensions( + dims, + [ + BandDimension( + name="bands", + bands=[ + Band("y", "yellow", 5), + Band("c", None, 8), + Band("m", "magenta", None), + ], + ) + ], + ) + assert logs == ["No cube:dimensions metadata", "Assuming name 'bands' for anonymous band dimension."] def test_get_dimensions_no_band_dimension_with_eo_bands(): logs = [] - dims = CollectionMetadata._parse_dimensions({ - "cube:dimensions": { - "x": {"type": "spatial", "extent": [-10, 10]}, - }, - "summaries": { - "eo:bands": [ - {"name": "y", "common_name": "yellow", "center_wavelength": 5}, - {"name": "c", "center_wavelength": 8}, - {"name": "m", "common_name": "magenta"}, - ] + dims = CollectionMetadata._parse_dimensions( + { + "cube:dimensions": { + "x": {"type": "spatial", "extent": [-10, 10]}, + }, + "summaries": { + "eo:bands": [ + {"name": "y", "common_name": "yellow", "center_wavelength": 5}, + {"name": "c", "center_wavelength": 8}, + {"name": "m", "common_name": "magenta"}, + ] + }, }, - }, complain=logs.append) - assert_same_dimensions(dims, [ - SpatialDimension(name="x", extent=[-10, 10]), - ]) + complain=logs.append, + ) + assert_same_dimensions( + dims, + [ + SpatialDimension(name="x", extent=[-10, 10]), + ], + ) assert logs == ["No 'bands' dimension in 'cube:dimensions' while having 'eo:bands' or 'raster:bands'"] def test_get_dimensions_multiple_band_dimensions_with_eo_bands(): logs = [] - dims = CollectionMetadata._parse_dimensions({ - "cube:dimensions": { - "x": {"type": "spatial", "extent": [-10, 10]}, - "spectral": {"type": "bands", "values": ["alpha", "beta"]}, - "bands": {"type": "bands", "values": ["r", "g", "b"]}, - }, - "summaries": { - "eo:bands": [ - {"name": "zu", "common_name": "foo"}, - ] + dims = CollectionMetadata._parse_dimensions( + { + "cube:dimensions": { + "x": {"type": "spatial", "extent": [-10, 10]}, + "spectral": {"type": "bands", "values": ["alpha", "beta"]}, + "bands": {"type": "bands", "values": ["r", "g", "b"]}, + }, + "summaries": { + "eo:bands": [ + {"name": "zu", "common_name": "foo"}, + ] + }, }, - }, complain=logs.append) - assert_same_dimensions(dims, [ - SpatialDimension(name="x", extent=[-10, 10]), - BandDimension(name="spectral", bands=[Band("alpha", None, None), Band("beta", None, None), ]), - BandDimension(name="bands", bands=[Band("r", None, None), Band("g", None, None), Band("b", None, None), ]), - ]) + complain=logs.append, + ) + assert_same_dimensions( + dims, + [ + SpatialDimension(name="x", extent=[-10, 10]), + BandDimension( + name="spectral", + bands=[ + Band("alpha", None, None), + Band("beta", None, None), + ], + ), + BandDimension( + name="bands", + bands=[ + Band("r", None, None), + Band("g", None, None), + Band("b", None, None), + ], + ), + ], + ) assert logs == ["Multiple dimensions of type 'bands'"] -@pytest.mark.parametrize("spec", [ - # API 0.4 style - { - "properties": { - "cube:dimensions": { - "x": {"type": "spatial", "axis": "x"}, - "b": {"type": "bands", "values": ["foo", "bar"]} - }}}, - # API 1.0 style - { - "cube:dimensions": { - "x": {"type": "spatial", "axis": "x"}, - "b": {"type": "bands", "values": ["foo", "bar"]} - }}, -]) -def test_metadata_bands_dimension_cube_dimensions(spec): +@pytest.mark.parametrize( + "spec", + [ + # API 0.4 style + { + "properties": { + "cube:dimensions": { + "x": {"type": "spatial", "axis": "x"}, + "b": {"type": "bands", "values": ["foo", "bar"]}, + } + } + }, + # API 1.0 style + {"cube:dimensions": {"x": {"type": "spatial", "axis": "x"}, "b": {"type": "bands", "values": ["foo", "bar"]}}}, + ], +) +def test_collectionmetadata_bands_dimension_cube_dimensions(spec): metadata = CollectionMetadata(spec) assert metadata.band_dimension.name == "b" - assert metadata.bands == [ - Band("foo", None, None), - Band("bar", None, None) - ] + assert metadata.bands == [Band("foo", None, None), Band("bar", None, None)] assert metadata.band_names == ["foo", "bar"] assert metadata.band_common_names == [None, None] -def test_metadata_bands_dimension_no_band_dimensions(): - metadata = CollectionMetadata({ - "cube:dimensions": { - "x": {"type": "spatial", "axis": "x"}, +def test_cubemetdata_bands_dimension(): + metadata = CubeMetadata( + dimensions=[ + SpatialDimension(name="x", extent=None), + BandDimension(name="b", bands=[Band("foo"), Band("bar")]), + ] + ) + assert metadata.band_dimension.name == "b" + assert metadata.bands == [Band("foo", None, None), Band("bar", None, None)] + assert metadata.band_names == ["foo", "bar"] + assert metadata.band_common_names == [None, None] + + +def test_collectionmetadata_bands_dimension_no_band_dimensions(): + metadata = CollectionMetadata( + { + "cube:dimensions": { + "x": {"type": "spatial", "axis": "x"}, + } } - }) + ) + with pytest.raises(MetadataException, match="No band dimension"): + metadata.band_dimension + with pytest.raises(MetadataException, match="No band dimension"): + metadata.bands + with pytest.raises(MetadataException, match="No band dimension"): + metadata.band_common_names + with pytest.raises(MetadataException, match="No band dimension"): + metadata.get_band_index("red") + with pytest.raises(MetadataException, match="No band dimension"): + metadata.filter_bands(["red"]) + + +def test_cubemetadata_bands_dimension_no_band_dimensions(): + metadata = CubeMetadata(dimensions=[SpatialDimension(name="x", extent=None)]) with pytest.raises(MetadataException, match="No band dimension"): metadata.band_dimension with pytest.raises(MetadataException, match="No band dimension"): @@ -383,103 +496,106 @@ def test_metadata_bands_dimension_no_band_dimensions(): metadata.filter_bands(["red"]) -@pytest.mark.parametrize("spec", [ - # API 0.4 style - { - "properties": { - "eo:bands": [ - {"name": "foo", "common_name": "F00", "center_wavelength": 0.543}, - {"name": "bar"} - ] - }}, - # API 1.0 style - { - "summaries": { - "eo:bands": [ - {"name": "foo", "common_name": "F00", "center_wavelength": 0.543}, - {"name": "bar"} - ] - }}, -]) +@pytest.mark.parametrize( + "spec", + [ + # API 0.4 style + { + "properties": { + "eo:bands": [{"name": "foo", "common_name": "F00", "center_wavelength": 0.543}, {"name": "bar"}] + } + }, + # API 1.0 style + { + "summaries": { + "eo:bands": [{"name": "foo", "common_name": "F00", "center_wavelength": 0.543}, {"name": "bar"}] + } + }, + ], +) def test_metadata_bands_dimension_eo_bands(spec): metadata = CollectionMetadata(spec) assert metadata.band_dimension.name == "bands" - assert metadata.bands == [ - Band("foo", "F00", 0.543), - Band("bar", None, None) - ] + assert metadata.bands == [Band("foo", "F00", 0.543), Band("bar", None, None)] assert metadata.band_names == ["foo", "bar"] assert metadata.band_common_names == ["F00", None] -@pytest.mark.parametrize("spec", [ - # API 0.4 style - { - "properties": { +@pytest.mark.parametrize( + "spec", + [ + # API 0.4 style + { + "properties": { + "cube:dimensions": { + "x": {"type": "spatial", "axis": "x"}, + "b": {"type": "bands", "values": ["foo", "bar"]}, + }, + "eo:bands": [{"name": "foo", "common_name": "F00", "center_wavelength": 0.543}, {"name": "bar"}], + } + }, + # API 1.0 style + { "cube:dimensions": { "x": {"type": "spatial", "axis": "x"}, - "b": {"type": "bands", "values": ["foo", "bar"]} + "b": {"type": "bands", "values": ["foo", "bar"]}, + }, + "summaries": { + "eo:bands": [{"name": "foo", "common_name": "F00", "center_wavelength": 0.543}, {"name": "bar"}] }, - "eo:bands": [ - {"name": "foo", "common_name": "F00", "center_wavelength": 0.543}, - {"name": "bar"} - ] - } - }, - # API 1.0 style - { - "cube:dimensions": { - "x": {"type": "spatial", "axis": "x"}, - "b": {"type": "bands", "values": ["foo", "bar"]} }, - "summaries": { - "eo:bands": [ - {"name": "foo", "common_name": "F00", "center_wavelength": 0.543}, - {"name": "bar"} - ] - } - }, -]) + ], +) def test_metadata_bands_dimension(spec): metadata = CollectionMetadata(spec) assert metadata.band_dimension.name == "b" - assert metadata.bands == [ - Band("foo", "F00", 0.543), - Band("bar", None, None) - ] + assert metadata.bands == [Band("foo", "F00", 0.543), Band("bar", None, None)] assert metadata.band_names == ["foo", "bar"] assert metadata.band_common_names == ["F00", None] -def test_metadata_reduce_dimension(): - metadata = CollectionMetadata({ - "cube:dimensions": { - "x": {"type": "spatial"}, - "b": {"type": "bands", "values": ["red", "green"]} - } - }) +def test_collectionmetadata_reduce_dimension(): + metadata = CollectionMetadata( + {"cube:dimensions": {"x": {"type": "spatial"}, "b": {"type": "bands", "values": ["red", "green"]}}} + ) reduced = metadata.reduce_dimension("b") assert set(metadata.dimension_names()) == {"x", "b"} assert set(reduced.dimension_names()) == {"x"} -def test_metadata_reduce_dimension_invalid_name(): - metadata = CollectionMetadata({ - "cube:dimensions": { - "x": {"type": "spatial"}, - "b": {"type": "bands", "values": ["red", "green"]} - } - }) +def test_cubemetadata_reduce_dimension(): + metadata = CubeMetadata( + dimensions=[ + SpatialDimension(name="x", extent=None), + BandDimension(name="b", bands=[Band("red"), Band("green")]), + ] + ) + reduced = metadata.reduce_dimension("b") + assert set(metadata.dimension_names()) == {"x", "b"} + assert set(reduced.dimension_names()) == {"x"} + + +def test_collectionmetadata_reduce_dimension_invalid_name(): + metadata = CollectionMetadata( + {"cube:dimensions": {"x": {"type": "spatial"}, "b": {"type": "bands", "values": ["red", "green"]}}} + ) with pytest.raises(ValueError): metadata.reduce_dimension("y") -def test_metadata_add_band_dimension(): - metadata = CollectionMetadata({ - "cube:dimensions": { - "t": {"type": "temporal"} - } - }) +def test_cubemetadata_reduce_dimension_invalid_name(): + metadata = CubeMetadata( + dimensions=[ + SpatialDimension(name="x", extent=None), + BandDimension(name="b", bands=[Band("red"), Band("green")]), + ] + ) + with pytest.raises(ValueError): + metadata.reduce_dimension("y") + + +def test_collectionmetadata_add_band_dimension(): + metadata = CollectionMetadata({"cube:dimensions": {"t": {"type": "temporal"}}}) new = metadata.add_dimension("layer", "red", "bands") assert metadata.dimension_names() == ["t"] @@ -491,23 +607,35 @@ def test_metadata_add_band_dimension(): assert new.band_names == ["red"] -def test_metadata_add_band_dimension_duplicate(): - metadata = CollectionMetadata({ - "cube:dimensions": { - "t": {"type": "temporal"} - } - }) +def test_cubemetadata_add_band_dimension(): + metadata = CubeMetadata(dimensions=[TemporalDimension(name="t", extent=None)]) + new = metadata.add_dimension("layer", "red", "bands") + + assert metadata.dimension_names() == ["t"] + assert not metadata.has_band_dimension() + + assert new.has_band_dimension() + assert new.dimension_names() == ["t", "layer"] + assert new.band_dimension.name == "layer" + assert new.band_names == ["red"] + + +def test_collectionmetadata_add_band_dimension_duplicate(): + metadata = CollectionMetadata({"cube:dimensions": {"t": {"type": "temporal"}}}) metadata = metadata.add_dimension("layer", "red", "bands") with pytest.raises(DimensionAlreadyExistsException, match="Dimension with name 'layer' already exists"): _ = metadata.add_dimension("layer", "red", "bands") -def test_metadata_add_temporal_dimension(): - metadata = CollectionMetadata({ - "cube:dimensions": { - "x": {"type": "spatial"} - } - }) +def test_cubemetadata_add_band_dimension_dublicate(): + metadata = CubeMetadata(dimensions=[TemporalDimension(name="t", extent=None)]) + metadata = metadata.add_dimension("layer", "red", "bands") + with pytest.raises(DimensionAlreadyExistsException, match="Dimension with name 'layer' already exists"): + _ = metadata.add_dimension("layer", "red", "bands") + + +def test_collectionmetadata_add_temporal_dimension(): + metadata = CollectionMetadata({"cube:dimensions": {"x": {"type": "spatial"}}}) new = metadata.add_dimension("date", "2020-05-15", "temporal") assert metadata.dimension_names() == ["x"] @@ -519,24 +647,42 @@ def test_metadata_add_temporal_dimension(): assert new.temporal_dimension.extent == ["2020-05-15", "2020-05-15"] -def test_metadata_add_temporal_dimension_duplicate(): - metadata = CollectionMetadata({ - "cube:dimensions": { - "x": {"type": "spatial"} - } - }) +def test_cubemetadata_add_temporal_dimension(): + metadata = CubeMetadata(dimensions=[SpatialDimension(name="x", extent=None)]) + new = metadata.add_dimension("date", "2020-05-15", "temporal") + + assert metadata.dimension_names() == ["x"] + assert not metadata.has_temporal_dimension() + + assert new.has_temporal_dimension() + assert new.dimension_names() == ["x", "date"] + assert new.temporal_dimension.name == "date" + assert new.temporal_dimension.extent == ["2020-05-15", "2020-05-15"] + + +def test_collectionmetadata_add_temporal_dimension_duplicate(): + metadata = CollectionMetadata({"cube:dimensions": {"x": {"type": "spatial"}}}) metadata = metadata.add_dimension("date", "2020-05-15", "temporal") with pytest.raises(DimensionAlreadyExistsException, match="Dimension with name 'date' already exists"): _ = metadata.add_dimension("date", "2020-05-15", "temporal") -def test_metadata_drop_dimension(): - metadata = CollectionMetadata({ - "cube:dimensions": { - "t": {"type": "temporal"}, - "b": {"type": "bands", "values": ["red", "green"]}, +def test_cubemetadata_add_temporal_dimension_duplicate(): + metadata = CubeMetadata(dimensions=[SpatialDimension(name="x", extent=None)]) + metadata = metadata.add_dimension("date", "2020-05-15", "temporal") + with pytest.raises(DimensionAlreadyExistsException, match="Dimension with name 'date' already exists"): + _ = metadata.add_dimension("date", "2020-05-15", "temporal") + + +def test_collectionmetadata_drop_dimension(): + metadata = CollectionMetadata( + { + "cube:dimensions": { + "t": {"type": "temporal"}, + "b": {"type": "bands", "values": ["red", "green"]}, + } } - }) + ) new = metadata.drop_dimension("t") assert metadata.dimension_names() == ["t", "b"] @@ -552,14 +698,36 @@ def test_metadata_drop_dimension(): metadata.drop_dimension("x") -def test_metadata_subclass(): +def test_cubemetadata_drop_dimension(): + metadata = CubeMetadata( + dimensions=[ + TemporalDimension(name="t", extent=None), + BandDimension(name="b", bands=[Band("red"), Band("green")]), + ] + ) + + new = metadata.drop_dimension("t") + assert metadata.dimension_names() == ["t", "b"] + assert new.dimension_names() == ["b"] + assert new.band_dimension.band_names == ["red", "green"] + + new = metadata.drop_dimension("b") + assert metadata.dimension_names() == ["t", "b"] + assert new.dimension_names() == ["t"] + assert new.temporal_dimension.name == "t" + + with pytest.raises(ValueError): + metadata.drop_dimension("x") + + +def test_collectionmetadata_subclass(): class MyCollectionMetadata(CollectionMetadata): def __init__(self, metadata: dict, dimensions: List[Dimension] = None, bbox=None): super().__init__(metadata=metadata, dimensions=dimensions) self.bbox = bbox def _clone_and_update( - self, metadata: dict = None, dimensions: List[Dimension] = None, bbox=None, **kwargs + self, metadata: dict = None, dimensions: List[Dimension] = None, bbox=None, **kwargs ) -> MyCollectionMetadata: return super()._clone_and_update(metadata=metadata, dimensions=dimensions, bbox=bbox or self.bbox, **kwargs) @@ -583,3 +751,34 @@ def filter_bbox(self, bbox): assert isinstance(new, MyCollectionMetadata) assert orig.bbox is None assert new.bbox == (1, 2, 3, 4) + + +def test_cubemetadata_subclass(): + class MyCubeMetadata(CubeMetadata): + def __init__(self, dimensions: List[Dimension], bbox=None): + super().__init__(dimensions=dimensions) + self.bbox = bbox + + def _clone_and_update(self, dimensions: List[Dimension] = None, bbox=None, **kwargs) -> MyCubeMetadata: + return super()._clone_and_update(dimensions=dimensions, bbox=bbox or self.bbox, **kwargs) + + def filter_bbox(self, bbox): + return self._clone_and_update(bbox=bbox) + + orig = MyCubeMetadata([SpatialDimension(name="x", extent=None)]) + assert orig.bbox is None + + new = orig.add_dimension(name="layer", label="red", type="bands") + assert isinstance(new, MyCubeMetadata) + assert orig.bbox is None + assert new.bbox is None + + new = new.filter_bbox((1, 2, 3, 4)) + assert isinstance(new, MyCubeMetadata) + assert orig.bbox is None + assert new.bbox == (1, 2, 3, 4) + + new = new.add_dimension(name="time", label="2020", type="time") + assert isinstance(new, MyCubeMetadata) + assert orig.bbox is None + assert new.bbox == (1, 2, 3, 4)