From eeef1cdfb48c73e84ea31a005f27037cfab1146d Mon Sep 17 00:00:00 2001 From: John Bogovic Date: Mon, 2 Dec 2024 14:34:34 -0500 Subject: [PATCH] fix: export in NGFF v0.4 required axis order add a test --- .../n5/ij/N5ScalePyramidExporter.java | 64 ++++++++++-- .../saalfeldlab/n5/metadata/NgffTests.java | 99 +++++++++++++++++-- 2 files changed, 149 insertions(+), 14 deletions(-) diff --git a/src/main/java/org/janelia/saalfeldlab/n5/ij/N5ScalePyramidExporter.java b/src/main/java/org/janelia/saalfeldlab/n5/ij/N5ScalePyramidExporter.java index a6243bed..6c7a6d4e 100644 --- a/src/main/java/org/janelia/saalfeldlab/n5/ij/N5ScalePyramidExporter.java +++ b/src/main/java/org/janelia/saalfeldlab/n5/ij/N5ScalePyramidExporter.java @@ -532,12 +532,7 @@ public & NativeType, M extends N5DatasetMetadata, N ex // get the image to save final RandomAccessibleInterval baseImg = getBaseImage(); - final M baseMetadata; - if (impMeta != null) - baseMetadata = (M)impMeta.readMetadata(image); - else - baseMetadata = null; - + final M baseMetadata = initializeBaseMetadata(); currentChannelMetadata = copyMetadata(baseMetadata); M currentMetadata; @@ -561,6 +556,7 @@ public & NativeType, M extends N5DatasetMetadata, N ex final double[] currentResolution = new double[nd]; System.arraycopy(baseResolution, 0, currentResolution, 0, nd); + // TODO here final N multiscaleMetadata = initializeMultiscaleMetadata((M)currentMetadata, channelDataset); currentTranslation = new double[nd]; @@ -629,6 +625,53 @@ public & NativeType, M extends N5DatasetMetadata, N ex n5.close(); } + @SuppressWarnings("unchecked") + protected M initializeBaseMetadata() { + + M baseMetadata = null; + if (impMeta != null) { + try { + baseMetadata = (M) impMeta.readMetadata(image); + } catch (IOException e) { + } + } + + if (impMeta instanceof NgffToImagePlus) { + + /* + * ImagePlus axes need to be permuted before conversion to ngff (i.e. from XYCZT + * to XYZCT) The data are permuted elsewhere, here we ensure the metadata + * reflect that chagnge + */ + final NgffSingleScaleAxesMetadata ngffMeta = (NgffSingleScaleAxesMetadata) baseMetadata; + baseMetadata = (M) new NgffSingleScaleAxesMetadata(ngffMeta.getPath(), ngffMeta.getScale(), + ngffMeta.getTranslation(), permuteAxesForNgff(ngffMeta.getAxes()), ngffMeta.getAttributes()); + } + + return baseMetadata; + } + + protected Axis[] permuteAxesForNgff(final Axis[] axes) { + + boolean hasC = false; + boolean hasZ = false; + boolean hasT = false; + for (int i = 0; i < axes.length; i++) { + hasC = hasC || axes[i].getName().equals("c"); + hasZ = hasZ || axes[i].getName().equals("z"); + hasT = hasT || axes[i].getName().equals("t"); + } + + if (hasC && hasZ) { + if (hasT) + return new Axis[] { axes[0], axes[1], axes[3], axes[2], axes[4] }; + else + return new Axis[] { axes[0], axes[1], axes[3], axes[2] }; + } + + return axes; + } + protected void initializeDataset() { dataset = image.getShortTitle(); @@ -936,10 +979,16 @@ protected & NativeType, M extends N5DatasetMetadata> L // some metadata styles never split channels, return input image in that // case if (metadataStyle.equals(NONE) || metadataStyle.equals(N5Importer.MetadataCustomKey) || - metadataStyle.equals(N5Importer.MetadataOmeZarrKey) || metadataStyle.equals(N5Importer.MetadataImageJKey)) { return Collections.singletonList(img); } + else if (metadataStyle.equals(N5Importer.MetadataOmeZarrKey)) { + + if (image.getNChannels() > 1 && image.getNSlices() > 1) + return Collections.singletonList(Views.permute(img, 2, 3)); + else + return Collections.singletonList(img); + } // otherwise, split channels final ArrayList> channels = new ArrayList<>(); @@ -962,6 +1011,7 @@ protected & NativeType, M extends N5DatasetMetadata> L // make a 4d image in order XYZT channelImg = Views.permute(Views.addDimension(channelImg, 0, 0), 2, 3); } + channels.add(channelImg); } diff --git a/src/test/java/org/janelia/saalfeldlab/n5/metadata/NgffTests.java b/src/test/java/org/janelia/saalfeldlab/n5/metadata/NgffTests.java index 90734b7a..513d0e4c 100644 --- a/src/test/java/org/janelia/saalfeldlab/n5/metadata/NgffTests.java +++ b/src/test/java/org/janelia/saalfeldlab/n5/metadata/NgffTests.java @@ -1,33 +1,57 @@ package org.janelia.saalfeldlab.n5.metadata; +import static org.junit.Assert.assertArrayEquals; +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertTrue; import static org.junit.Assert.fail; +import java.io.File; import java.io.IOException; +import java.nio.file.Files; +import java.util.Arrays; +import org.janelia.saalfeldlab.n5.DatasetAttributes; import org.janelia.saalfeldlab.n5.N5Exception; import org.janelia.saalfeldlab.n5.N5FSReader; +import org.janelia.saalfeldlab.n5.N5Reader; +import org.janelia.saalfeldlab.n5.ij.N5Importer; +import org.janelia.saalfeldlab.n5.ij.N5ScalePyramidExporter; +import org.janelia.saalfeldlab.n5.universe.N5Factory; import org.janelia.saalfeldlab.n5.universe.metadata.NgffMultiScaleGroupAttributes; import org.janelia.saalfeldlab.n5.universe.metadata.NgffMultiScaleGroupAttributes.MultiscaleDataset; +import org.janelia.saalfeldlab.n5.universe.metadata.axes.Axis; import org.junit.Assert; -import org.junit.Before; +import org.junit.BeforeClass; import org.junit.Test; + +import ij.ImagePlus; +import ij.gui.NewImage; + public class NgffTests { - private N5FSReader n5; + private final String n5Root = "src/test/resources/ngff.n5"; + + private static File baseDir; + + @BeforeClass + public static void setup() { - @Before - public void setUp() throws IOException { + try { + baseDir = Files.createTempDirectory("ngff-tests-").toFile(); + baseDir.deleteOnExit(); - final String n5Root = "src/test/resources/ngff.n5"; - n5 = new N5FSReader(n5Root); + } catch (IOException e) { + e.printStackTrace(); + } } @Test public void testNgffGroupAttributeParsing() { final double eps = 1e-9; - try { + try( final N5FSReader n5 = new N5FSReader(n5Root) ) { + NgffMultiScaleGroupAttributes[] multiscales = n5.getAttribute("ngff_grpAttributes", "multiscales", NgffMultiScaleGroupAttributes[].class); Assert.assertEquals("one set of multiscales", 1, multiscales.length); @@ -51,4 +75,65 @@ public void testNgffGroupAttributeParsing() { } } + @Test + public void testNgffExportAxisOrder() { + + testNgfffAxisOrder("xyczt", new int[] { 10, 8, 6, 4, 2 }); + + testNgfffAxisOrder("xyzt", new int[] { 10, 8, 1, 4, 2 }); + testNgfffAxisOrder("xyct", new int[] { 10, 8, 6, 1, 2 }); + testNgfffAxisOrder("xycz", new int[] { 10, 8, 6, 4, 1 }); + + testNgfffAxisOrder("xyc", new int[] { 10, 8, 6, 1, 1 }); + testNgfffAxisOrder("xyz", new int[] { 10, 8, 1, 4, 1 }); + testNgfffAxisOrder("xyt", new int[] { 10, 8, 1, 1, 2 }); + } + + public void testNgfffAxisOrder(final String dataset, int[] size) { + + final int nx = size[0]; + final int ny = size[1]; + final int nc = size[2]; + final int nz = size[3]; + final int nt = size[4]; + + final String metadataType = N5Importer.MetadataOmeZarrKey; + final String compressionType = N5ScalePyramidExporter.RAW_COMPRESSION; + + final ImagePlus imp = NewImage.createImage("test", nx, ny, nz * nc * nt, 8, NewImage.FILL_BLACK); + imp.setDimensions(nc, nz, nt); + + final N5ScalePyramidExporter writer = new N5ScalePyramidExporter(); + writer.setOptions(imp, baseDir.getAbsolutePath(), dataset, N5ScalePyramidExporter.ZARR_FORMAT, "64", false, + N5ScalePyramidExporter.DOWN_SAMPLE, metadataType, compressionType); + writer.run(); + + final long[] expectedDims = Arrays.stream(new long[] { nx, ny, nz, nc, nt }).filter(x -> x > 1).toArray(); + + try (final N5Reader n5 = new N5Factory().openReader(baseDir.getAbsolutePath())) { + + assertTrue(n5.exists(dataset)); + assertTrue(n5.datasetExists(dataset + "/s0")); + + final DatasetAttributes dsetAttrs = n5.getDatasetAttributes(dataset + "/s0"); + assertArrayEquals("dimensions", expectedDims, dsetAttrs.getDimensions()); + + int i = 0; + final Axis[] axes = n5.getAttribute(dataset, "multiscales[0]/axes", Axis[].class); + + if (nt > 1) + assertEquals("t", axes[i++].getName()); + + if (nc > 1) + assertEquals("c", axes[i++].getName()); + + if (nz > 1) + assertEquals("z", axes[i++].getName()); + + assertEquals("y", axes[i++].getName()); + assertEquals("x", axes[i++].getName()); + } + + } + }