Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

rename-wells CLI utility #232

Merged
merged 29 commits into from
Oct 3, 2024
Merged
Show file tree
Hide file tree
Changes from 13 commits
Commits
Show all changes
29 commits
Select commit Hold shift + click to select a range
91e7fdd
Updated rename wells functionality at CLI level, fixed style errors
Jul 7, 2024
d336cd8
CLI changes
Jul 9, 2024
e3f7029
fix click options
talonchandler Jul 9, 2024
4fb8b03
basic test for help message
talonchandler Jul 9, 2024
1f27a19
Updated decorator
Jul 11, 2024
66e5d12
Exception changes
Jul 11, 2024
75a9327
Updated rename wells and testing
Jul 12, 2024
f256cfe
Updated test
Jul 12, 2024
65a0394
Updated test
Jul 15, 2024
01132dc
test_rename_wells_basic
Jul 15, 2024
c7c42ed
Updated tests, converting well names back to original
Jul 15, 2024
452f8ed
Context manager update
Jul 15, 2024
c753a87
well-mapping.csv
Jul 17, 2024
6851360
Updating row and column indices
Jul 21, 2024
ce70316
Style changes
Jul 21, 2024
92e53dd
Merge branch 'main' into renamewells
talonchandler Sep 19, 2024
31888c6
rework API with `self.zgroup.move`
talonchandler Sep 19, 2024
5b59dec
test API
talonchandler Sep 19, 2024
fb0685a
simplify and document CLI
talonchandler Sep 19, 2024
410f482
test CLI w/ round trip
talonchandler Sep 19, 2024
5caa257
style
talonchandler Sep 19, 2024
d1f7639
make example typical
talonchandler Sep 19, 2024
2c0c083
move rename_wells to its own file
ieivanov Oct 1, 2024
eb40820
add checks for correct well names
ieivanov Oct 1, 2024
49740fe
Merge branch 'main' into pr/232
ieivanov Oct 1, 2024
f9b6847
add tests for invalid well names
ieivanov Oct 1, 2024
799ccfe
Merge branch 'main' into pr/232
ieivanov Oct 3, 2024
27cda8d
remove test_rename_wells deadline
ieivanov Oct 3, 2024
33a61a4
remove test_set_scale deadline
ieivanov Oct 3, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 4 additions & 0 deletions docs/examples/well-mapping.csv
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
0/0,0/A
0/1,0/B
0/2,0/C
0/3,0/D
68 changes: 68 additions & 0 deletions iohub/cli/cli.py
Original file line number Diff line number Diff line change
@@ -1,9 +1,11 @@
import csv
import pathlib

import click

from iohub._version import __version__
from iohub.convert import TIFFConverter
from iohub.ngff import open_ome_zarr
from iohub.reader import print_info

VERSION = __version__
Expand Down Expand Up @@ -87,3 +89,69 @@ def convert(input, output, grid_layout, chunks):
chunks=chunks,
)
converter()


@cli.command()
@click.help_option("-h", "--help")
@click.option(
"-i",
"--input",
"zarrfile",
type=click.Path(exists=True, file_okay=True, dir_okay=True),
required=True,
help="Path to the input Zarr file.",
)
@click.option(
"-c",
"--csv",
"csvfile",
type=click.File("r"),
required=True,
help="Path to the CSV file containing well names.",
)
def rename_wells(csvfile, zarrfile):
josephschull marked this conversation as resolved.
Show resolved Hide resolved
rename_wells_cli(csvfile, zarrfile)


def rename_wells_cli(csvfile, zarrfile):
josephschull marked this conversation as resolved.
Show resolved Hide resolved
"""Rename wells based on CSV file

The CSV file should have two columns: old_well_path and new_well_path.
"""

names = []

csvreader = csv.reader(csvfile)
for row in csvreader:
if len(row) != 2:
raise ValueError(
f"Invalid row format: {row}."
f"Each row must have two columns."
)
names.append([row[0], row[1]])

with open_ome_zarr(zarrfile, mode="a") as plate:
well_paths = [well.path for well in plate.metadata.wells]
print(f"Initial well paths: {well_paths}")

modified = []

for old_well_path, new_well_path in names:
for well in plate.metadata.wells:
if (
str(well.path) == str(old_well_path)
and well not in modified
):
print(f"Renaming {old_well_path} to {new_well_path}...")
try:
plate.rename_well(well, old_well_path, new_well_path)
modified.append(well)
print(
f"Well {old_well_path} renamed to {new_well_path}"
)
except ValueError as e:
click.echo(f"Error: {e}", err=True)

print(
f"Final well paths: {[well.path for well in plate.metadata.wells]}"
)
44 changes: 44 additions & 0 deletions iohub/ngff.py
Original file line number Diff line number Diff line change
Expand Up @@ -1565,6 +1565,50 @@ def positions(self) -> Generator[tuple[str, Position], None, None]:
for _, position in well.positions():
yield position.zgroup.path, position

def rename_well(
self,
well,
old_well_path: str,
josephschull marked this conversation as resolved.
Show resolved Hide resolved
new_well_path: str,
):
"""Rename a well.

Parameters
----------
old_well_path : str
Old name of well
new_well_path : str
New name of well
"""
old_row, old_column = old_well_path.split("/")
new_row, new_column = new_well_path.split("/")

well_paths = [well.path for well in self.metadata.wells]

if old_well_path not in well_paths:
raise ValueError(f"Well '{old_well_path}' not found in plate.")

elif old_well_path in well_paths:
well = next(
w for w in self.metadata.wells if w.path == old_well_path
)

if well.path == old_well_path:
well.path = new_well_path # update well metadata
zarr.storage.rename(
self.zgroup._store, old_well_path, new_well_path
) # update well paths

for column in self.metadata.columns:
if column.name == old_column:
column.name = new_column # update column metadata

for row in self.metadata.rows:
if row.name == old_row:
row.name = new_row # update row metadata

self.dump_meta()


def open_ome_zarr(
store_path: StrOrBytesPath,
Expand Down
69 changes: 66 additions & 3 deletions tests/cli/test_cli.py
Original file line number Diff line number Diff line change
@@ -1,17 +1,17 @@
import csv
import re
from unittest.mock import patch

from click.testing import CliRunner
import pytest
from click.testing import CliRunner

from iohub._version import __version__
from iohub.cli.cli import cli

from tests.conftest import (
hcs_ref,
mm2gamma_ome_tiffs,
ndtiff_v2_datasets,
ndtiff_v3_labeled_positions,
hcs_ref,
)


Expand Down Expand Up @@ -104,3 +104,66 @@ def test_cli_convert_ome_tiff(grid_layout, tmpdir):
result = runner.invoke(cli, cmd)
assert result.exit_code == 0, result.output
assert "Converting" in result.output


def test_rename_wells_help():
runner = CliRunner()
cmd = ["rename-wells"]
for option in ("-h", "--help"):
cmd.append(option)
result = runner.invoke(cli, cmd)
assert result.exit_code == 0
assert "containing well names" in result.output


def test_rename_wells(tmpdir):
runner = CliRunner()
test_csv = tmpdir / "well_names.csv"
csv_data = [
["B/03", "B/03test"],
]
with open(test_csv, mode="w", newline="") as csvfile:
writer = csv.writer(csvfile)
writer.writerows(csv_data)

cmd = ["rename-wells", "-i", hcs_ref, "-c", str(test_csv)]
result = runner.invoke(cli, cmd)

print(result.output)
assert result.exit_code == 0

final_well_paths = None

for line in result.output.split("\n"):
if line.startswith("Final well paths:"):
final_well_paths = eval(line.split(": ")[1])
break

assert (
final_well_paths is not None
), "Final well paths not found in the output"

new_well_paths = [row[1] for row in csv_data]
old_well_paths = [row[0] for row in csv_data]

for new_path in new_well_paths:
assert (
new_path in final_well_paths
), f"Expected {new_path} in final well paths"

for old_path in old_well_paths:
assert (
old_path not in final_well_paths
), f"Did not expect {old_path} in final well paths"

test_csv_2 = tmpdir / "well_names_2.csv"
csv_data = [
["B/03test", "B/03"],
]
with open(test_csv_2, mode="w", newline="") as csvfile:
writer = csv.writer(csvfile)
writer.writerows(csv_data)

cmd = ["rename-wells", "-i", hcs_ref, "-c", str(test_csv_2)]
result = runner.invoke(cli, cmd)
print(result.output)
Loading