-
Notifications
You must be signed in to change notification settings - Fork 68
/
runcellpose.py
753 lines (643 loc) · 31.3 KB
/
runcellpose.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
#################################
#
# Imports from useful Python libraries
#
#################################
import numpy
import os
import skimage
import importlib.metadata
import subprocess
import uuid
import shutil
import logging
import sys
#################################
#
# Imports from CellProfiler
#
##################################
from cellprofiler_core.image import Image
from cellprofiler_core.module.image_segmentation import ImageSegmentation
from cellprofiler_core.object import Objects
from cellprofiler_core.setting import Binary, ValidationError
from cellprofiler_core.setting.choice import Choice
from cellprofiler_core.setting.do_something import DoSomething
from cellprofiler_core.setting.subscriber import ImageSubscriber
from cellprofiler_core.preferences import get_default_output_directory
from cellprofiler_core.setting.text import (
Integer,
ImageName,
Directory,
Filename,
Float,
)
CUDA_LINK = "https://pytorch.org/get-started/locally/"
Cellpose_link = " https://doi.org/10.1038/s41592-020-01018-x"
Omnipose_link = "https://doi.org/10.1101/2021.11.03.467199"
LOGGER = logging.getLogger(__name__)
__doc__ = f"""\
RunCellpose
===========
**RunCellpose** uses a pre-trained machine learning model (Cellpose) to detect cells or nuclei in an image.
This module is useful for automating simple segmentation tasks in CellProfiler.
The module accepts greyscale input images and produces an object set. Probabilities can also be captured as an image.
Loading in a model will take slightly longer the first time you run it each session. When evaluating
performance you may want to consider the time taken to predict subsequent images.
This module now also supports Ominpose. Omnipose builds on Cellpose, for the purpose of **RunCellpose** it adds 2 additional
features: additional models; bact-omni and cyto2-omni which were trained using the Omnipose architechture, and bact
and the mask reconstruction algorithm for Omnipose that was created to solve over-segemnation of large cells; useful for bacterial cells,
but can be used for other arbitrary and anisotropic shapes. You can mix and match Omnipose models with Cellpose style masking or vice versa.
The module is compatible with Cellpose 1.0.2 >= 2.3.2. From the old version of the module the 'cells' model corresponds to 'cyto2' model.
You can run this module using Cellpose installed to the same Python environment as CellProfiler. Alternatively, you can
run this module using Cellpose in a Docker that the module will automatically download for you so you do not have to perform
any installation yourself.
To install Cellpose in your Python environment:
You'll want to run `pip install cellpose==2.3.2` on your CellProfiler Python environment to setup Cellpose. If you have an older version of Cellpose
run 'python -m pip install --force-reinstall -v cellpose==2.3.2'.
To use Omnipose models, and mask reconstruction method you'll want to install Omnipose 'pip install omnipose' and Cellpose version 1.0.2 'pip install cellpose==1.0.2'.
On the first time loading into CellProfiler, Cellpose will need to download some model files from the internet. This
may take some time. If you want to use a GPU to run the model, you'll need a compatible version of PyTorch and a
supported GPU. Instructions are avaiable at this link: {CUDA_LINK}
Stringer, C., Wang, T., Michaelos, M. et al. Cellpose: a generalist algorithm for cellular segmentation. Nat Methods 18, 100–106 (2021). {Cellpose_link}
Kevin J. Cutler, Carsen Stringer, Paul A. Wiggins, Joseph D. Mougous. Omnipose: a high-precision morphology-independent solution for bacterial cell segmentation. bioRxiv 2021.11.03.467199. {Omnipose_link}
============ ============ ===============
Supports 2D? Supports 3D? Respects masks?
============ ============ ===============
YES YES NO
============ ============ ===============
"""
"Select Cellpose Docker Image"
CELLPOSE_DOCKER_NO_PRETRAINED_v232 = "cellprofiler/runcellpose_no_pretrained:2.3.2"
CELLPOSE_DOCKER_IMAGE_WITH_PRETRAINED_v232 = "cellprofiler/runcellpose_with_pretrained:2.3.2"
CELLPOSE_DOCKER_NO_PRETRAINED_v220 = "cellprofiler/runcellpose_no_pretrained:2.2"
CELLPOSE_DOCKER_IMAGE_WITH_PRETRAINED_v220 = "cellprofiler/runcellpose_with_pretrained:2.2"
"Detection mode"
MODEL_NAMES = ['cyto','nuclei','tissuenet','livecell', 'cyto2', 'general',
'CP', 'CPx', 'TN1', 'TN2', 'TN3', 'LC1', 'LC2', 'LC3', 'LC4', 'custom']
class RunCellpose(ImageSegmentation):
category = "Object Processing"
module_name = "RunCellpose"
variable_revision_number = 6
doi = {
"Please cite the following when using RunCellPose:": "https://doi.org/10.1038/s41592-020-01018-x",
"If you are using Omnipose also cite the following:": "https://doi.org/10.1101/2021.11.03.467199",
}
def create_settings(self):
super(RunCellpose, self).create_settings()
self.rescale = Binary(
text="Rescale images before running Cellpose",
value=True,
doc="""\
Reminds the user that the normalization step will be performed to ensure suimilar segmentation behaviour in the RunCellpose
module and the Cellpose app.
"""
)
self.docker_or_python = Choice(
text="Run CellPose in docker or local python environment",
choices=["Docker", "Python"],
value="Docker",
doc="""\
If Docker is selected, ensure that Docker Desktop is open and running on your
computer. On first run of the RunCellpose plugin, the Docker container will be
downloaded. However, this slow downloading process will only have to happen
once.
If Python is selected, the Python environment in which CellProfiler and Cellpose
are installed will be used.
""",
)
self.docker_image = Choice(
text="Select Cellpose docker image",
choices=[CELLPOSE_DOCKER_IMAGE_WITH_PRETRAINED_v220, CELLPOSE_DOCKER_NO_PRETRAINED_v220,CELLPOSE_DOCKER_IMAGE_WITH_PRETRAINED_v232, CELLPOSE_DOCKER_NO_PRETRAINED_v232],
value=CELLPOSE_DOCKER_IMAGE_WITH_PRETRAINED_v232,
doc="""\
Select which Docker image to use for running Cellpose.
If you are not using a custom model, you can select
**"{CELLPOSE_DOCKER_IMAGE_WITH_PRETRAINED_v232}"** or "{CELLPOSE_DOCKER_IMAGE_WITH_PRETRAINED_v220}"**. If you are using a custom model,
you can use any of the available Dockers, but those with pretrained models will be slightly larger (~500 MB).
""".format(
**{"CELLPOSE_DOCKER_NO_PRETRAINED_v220": CELLPOSE_DOCKER_NO_PRETRAINED_v220,
"CELLPOSE_DOCKER_IMAGE_WITH_PRETRAINED_v220": CELLPOSE_DOCKER_IMAGE_WITH_PRETRAINED_v220,
"CELLPOSE_DOCKER_NO_PRETRAINED_v232": CELLPOSE_DOCKER_NO_PRETRAINED_v232,
"CELLPOSE_DOCKER_IMAGE_WITH_PRETRAINED_v232": CELLPOSE_DOCKER_IMAGE_WITH_PRETRAINED_v232}
),
)
self.expected_diameter = Integer(
text="Expected object diameter",
value=30,
minval=0,
doc="""\
The average diameter of the objects to be detected. Setting this to 0 will attempt to automatically detect object size.
Note that automatic diameter mode does not work when running on 3D images.
Cellpose models come with a pre-defined object diameter. Your image will be resized during detection to attempt to
match the diameter expected by the model. The default models have an expected diameter of ~16 pixels, if trying to
detect much smaller objects it may be more efficient to resize the image first using the Resize module.
""",
)
self.mode = Choice(
text="Detection mode",
choices=MODEL_NAMES,
value=MODEL_NAMES[0],
doc="""\
CellPose comes with models for detecting nuclei or cells. Alternatively, you can supply a custom-trained model
generated using the command line or Cellpose GUI. Custom models can be useful if working with unusual cell types.
""",
)
self.omni = Binary(
text="Use Omnipose for mask reconstruction",
value=False,
doc="""\
If enabled, use omnipose mask recontruction features will be used (Omnipose installation required and CellPose >= 1.0) """,
)
self.do_3D = Binary(
text="Use 3D",
value=False,
doc="""\
If enabled, 3D specific settings will be available.""",
)
self.use_gpu = Binary(
text="Use GPU",
value=False,
doc=f"""\
If enabled, Cellpose will attempt to run detection on your system's graphics card (GPU).
Note that you will need a CUDA-compatible GPU and correctly configured PyTorch version, see this link for details:
{CUDA_LINK}
If disabled or incorrectly configured, Cellpose will run on your CPU instead. This is much slower but more compatible
with different hardware setups.
Note that, particularly when in 3D mode, lack of GPU memory can become a limitation. If a model crashes you may need to
re-start CellProfiler to release GPU memory. Resizing large images prior to running them through the model can free up
GPU memory.
""",
)
self.use_averaging = Binary(
text="Use averaging",
value=False,
doc="""\
If enabled, CellPose will run it's 4 inbuilt models and take a consensus to determine the results. If disabled, only a
single model will be called to produce results. Disabling averaging is faster to run but less accurate.""",
)
self.invert = Binary(
text="Invert images",
value=False,
doc="""\
If enabled the image will be inverted and also normalized. For use with fluorescence images using bact model (bact model was trained on phase images""",
)
self.supply_nuclei = Binary(
text="Supply nuclei image as well?",
value=False,
doc="""
When detecting whole cells, you can provide a second image featuring a nuclear stain to assist
the model with segmentation. This can help to split touching cells.""",
)
self.nuclei_image = ImageSubscriber(
"Select the nuclei image",
doc="Select the image you want to use as the nuclear stain.",
)
self.save_probabilities = Binary(
text="Save probability image?",
value=False,
doc="""
If enabled, the probability scores from the model will be recorded as a new image.
Probability >0 is considered as being part of a cell.
You may want to use a higher threshold to manually generate objects.""",
)
self.probabilities_name = ImageName(
"Name the probability image",
"Probabilities",
doc="Enter the name you want to call the probability image produced by this module.",
)
self.model_directory = Directory(
"Location of the pre-trained model file",
doc=f"""\
*(Used only when using a custom pre-trained model)*
Select the location of the pre-trained CellPose model file that will be used for detection.""",
)
def get_directory_fn():
"""Get the directory for the rules file name"""
return self.model_directory.get_absolute_path()
def set_directory_fn(path):
dir_choice, custom_path = self.model_directory.get_parts_from_path(path)
self.model_directory.join_parts(dir_choice, custom_path)
self.model_file_name = Filename(
"Pre-trained model file name",
"cyto_0",
get_directory_fn=get_directory_fn,
set_directory_fn=set_directory_fn,
doc=f"""\
*(Used only when using a custom pre-trained model)*
This file can be generated by training a custom model withing the CellPose GUI or command line applications.""",
)
self.gpu_test = DoSomething(
"",
"Test GPU",
self.do_check_gpu,
doc=f"""\
Press this button to check whether a GPU is correctly configured.
If you have a dedicated GPU, a failed test usually means that either your GPU does not support deep learning or the
required dependencies are not installed.
If you have multiple GPUs on your system, this button will only test the first one.
""",
)
self.flow_threshold = Float(
text="Flow threshold",
value=0.4,
minval=0,
doc="""\
The flow_threshold parameter is the maximum allowed error of the flows for each mask. The default is flow_threshold=0.4.
Increase this threshold if cellpose is not returning as many masks as you’d expect.
Similarly, decrease this threshold if cellpose is returning too many ill-shaped masks
""",
)
self.cellprob_threshold = Float(
text="Cell probability threshold",
value=0.0,
minval=-6.0,
maxval=6.0,
doc=f"""\
Cell probability threshold (all pixels with probability above threshold kept for masks). Recommended default is 0.0.
Values vary from -6 to 6
""",
)
self.manual_GPU_memory_share = Float(
text="GPU memory share for each worker",
value=0.1,
minval=0.0000001,
maxval=1,
doc="""\
Fraction of the GPU memory share available to each worker. Value should be set such that this number times the number
of workers in each copy of CellProfiler times the number of copies of CellProfiler running (if applicable) is <1
""",
)
self.stitch_threshold = Float(
text="Stitch Threshold",
value=0.0,
minval=0,
doc=f"""\
There may be additional differences in YZ and XZ slices that make them unable to be used for 3D segmentation.
In those instances, you may want to turn off 3D segmentation (do_3D=False) and run instead with stitch_threshold>0.
Cellpose will create masks in 2D on each XY slice and then stitch them across slices if the IoU between the mask on the current slice and the next slice is greater than or equal to the stitch_threshold.
""",
)
self.min_size = Integer(
text="Minimum size",
value=15,
minval=-1,
doc="""\
Minimum number of pixels per mask, can turn off by setting value to -1
""",
)
self.remove_edge_masks = Binary(
text="Remove objects that are touching the edge?",
value=True,
doc="""
If you do not want to include any object masks that are not in full view in the image, you can have the masks that have pixels touching the the edges removed.
The default is set to "Yes".
""",
)
self.probability_rescale_setting = Binary(
text="Rescale probability map?",
value=True,
doc="""
Activate to rescale probability map to 0-255 (which matches the scale used when running this module from Docker)
""",
)
def settings(self):
return [
self.x_name,
self.rescale,
self.docker_or_python,
self.docker_image,
self.expected_diameter,
self.mode,
self.y_name,
self.use_gpu,
self.use_averaging,
self.supply_nuclei,
self.nuclei_image,
self.save_probabilities,
self.probabilities_name,
self.model_directory,
self.model_file_name,
self.flow_threshold,
self.cellprob_threshold,
self.manual_GPU_memory_share,
self.stitch_threshold,
self.do_3D,
self.min_size,
self.omni,
self.invert,
self.remove_edge_masks,
self.probability_rescale_setting,
]
def visible_settings(self):
vis_settings = [self.rescale, self.docker_or_python]
if self.docker_or_python.value == "Docker":
vis_settings += [self.docker_image]
vis_settings += [self.mode, self.x_name]
if self.docker_or_python.value == "Python":
vis_settings += [self.omni]
if self.mode.value != "nuclei":
vis_settings += [self.supply_nuclei]
if self.supply_nuclei.value:
vis_settings += [self.nuclei_image]
if self.mode.value == "custom":
vis_settings += [
self.model_directory,
self.model_file_name,
]
vis_settings += [
self.expected_diameter,
self.cellprob_threshold,
self.min_size,
self.flow_threshold,
self.y_name,
self.invert,
self.save_probabilities,
]
vis_settings += [self.do_3D, self.stitch_threshold, self.remove_edge_masks]
if self.do_3D.value:
vis_settings.remove(self.stitch_threshold)
if self.save_probabilities.value:
vis_settings += [self.probabilities_name]
if self.docker_or_python.value == 'Python':
vis_settings += [self.probability_rescale_setting]
vis_settings += [self.use_averaging, self.use_gpu]
if self.docker_or_python.value == 'Python':
if self.use_gpu.value:
vis_settings += [self.gpu_test, self.manual_GPU_memory_share]
return vis_settings
def validate_module(self, pipeline):
"""If using custom model, validate the model file opens and works"""
from cellpose import models
if self.mode.value == "custom":
model_file = self.model_file_name.value
model_directory = self.model_directory.get_absolute_path()
model_path = os.path.join(model_directory, model_file)
try:
open(model_path)
except:
raise ValidationError(
"Failed to load custom file: %s " % model_path,
self.model_file_name,
)
if self.docker_or_python.value == "Python":
try:
model = models.CellposeModel(pretrained_model=model_path, gpu=self.use_gpu.value)
except:
raise ValidationError(
"Failed to load custom model: %s "
% model_path, self.model_file_name,
)
def run(self, workspace):
x_name = self.x_name.value
y_name = self.y_name.value
images = workspace.image_set
x = images.get_image(x_name)
dimensions = x.dimensions
x_data = x.pixel_data
if self.rescale.value:
rescale_x = x_data.copy()
x01 = numpy.percentile(rescale_x, 1)
x99 = numpy.percentile(rescale_x, 99)
x_data = numpy.clip((rescale_x - x01) / (x99 - x01), a_min=0, a_max=1)
anisotropy = 0.0
if self.do_3D.value:
anisotropy = x.spacing[0] / x.spacing[1]
diam = self.expected_diameter.value if self.expected_diameter.value > 0 else None
if x.multichannel:
raise ValueError(
"Color images are not currently supported. Please provide greyscale images."
)
if self.mode.value != "nuclei" and self.supply_nuclei.value:
nuc_image = images.get_image(self.nuclei_image.value)
# CellPose expects RGB, we'll have a blank red channel, cells in green and nuclei in blue.
if self.do_3D.value:
x_data = numpy.stack(
(numpy.zeros_like(x_data), x_data, nuc_image.pixel_data), axis=1
)
else:
x_data = numpy.stack(
(numpy.zeros_like(x_data), x_data, nuc_image.pixel_data), axis=-1
)
channels = [2, 3]
else:
channels = [0, 0]
if self.docker_or_python.value == "Python":
from cellpose import models, io, core, utils
self.cellpose_ver = importlib.metadata.version('cellpose')
if float(self.cellpose_ver[0:3]) >= 0.6 and int(self.cellpose_ver[0])<2:
if self.mode.value != 'custom':
model = models.Cellpose(model_type= self.mode.value,
gpu=self.use_gpu.value)
else:
model_file = self.model_file_name.value
model_directory = self.model_directory.get_absolute_path()
model_path = os.path.join(model_directory, model_file)
model = models.CellposeModel(pretrained_model=model_path, gpu=self.use_gpu.value)
else:
if self.mode.value != 'custom':
model = models.CellposeModel(model_type= self.mode.value,
gpu=self.use_gpu.value)
else:
model_file = self.model_file_name.value
model_directory = self.model_directory.get_absolute_path()
model_path = os.path.join(model_directory, model_file)
model = models.CellposeModel(pretrained_model=model_path, gpu=self.use_gpu.value)
if self.use_gpu.value and model.torch:
from torch import cuda
cuda.set_per_process_memory_fraction(self.manual_GPU_memory_share.value)
try:
if float(self.cellpose_ver[0:3]) >= 0.7 and int(self.cellpose_ver[0])<2:
y_data, flows, *_ = model.eval(
x_data,
channels=channels,
diameter=diam,
net_avg=self.use_averaging.value,
do_3D=self.do_3D.value,
anisotropy=anisotropy,
flow_threshold=self.flow_threshold.value,
cellprob_threshold=self.cellprob_threshold.value,
stitch_threshold=self.stitch_threshold.value,
min_size=self.min_size.value,
omni=self.omni.value,
invert=self.invert.value,
)
else:
y_data, flows, *_ = model.eval(
x_data,
channels=channels,
diameter=diam,
net_avg=self.use_averaging.value,
do_3D=self.do_3D.value,
anisotropy=anisotropy,
flow_threshold=self.flow_threshold.value,
cellprob_threshold=self.cellprob_threshold.value,
stitch_threshold=self.stitch_threshold.value,
min_size=self.min_size.value,
invert=self.invert.value,
)
if self.remove_edge_masks:
y_data = utils.remove_edge_masks(y_data)
except Exception as a:
print(f"Unable to create masks. Check your module settings. {a}")
finally:
if self.use_gpu.value and model.torch:
# Try to clear some GPU memory for other worker processes.
try:
cuda.empty_cache()
except Exception as e:
print(f"Unable to clear GPU memory. You may need to restart CellProfiler to change models. {e}")
elif self.docker_or_python.value == "Docker":
# Define how to call docker
docker_path = "docker" if sys.platform.lower().startswith("win") else "/usr/local/bin/docker"
# Create a UUID for this run
unique_name = str(uuid.uuid4())
# Directory that will be used to pass images to the docker container
temp_dir = os.path.join(get_default_output_directory(), ".cellprofiler_temp", unique_name)
temp_img_dir = os.path.join(temp_dir, "img")
os.makedirs(temp_dir, exist_ok=True)
os.makedirs(temp_img_dir, exist_ok=True)
temp_img_path = os.path.join(temp_img_dir, unique_name+".tiff")
if self.mode.value == "custom":
model_file = self.model_file_name.value
model_directory = self.model_directory.get_absolute_path()
model_path = os.path.join(model_directory, model_file)
temp_model_dir = os.path.join(temp_dir, "model")
os.makedirs(temp_model_dir, exist_ok=True)
# Copy the model
shutil.copy(model_path, os.path.join(temp_model_dir, model_file))
# Save the image to the Docker mounted directory
skimage.io.imsave(temp_img_path, x_data)
cmd = [docker_path, 'run', '--rm', '-v', f'{temp_dir}:/data', self.docker_image.value]
if self.use_gpu.value:
cmd += ['--gpus', 'all']
cmd += ['cellpose', '--verbose', '--dir', '/data/img', '--pretrained_model']
if self.mode.value !='custom':
cmd += [self.mode.value]
else:
cmd += ['/data/model/' + model_file]
cmd += ['--chan', str(channels[0]), '--chan2', str(channels[1]), '--diameter', str(diam)]
if self.use_averaging.value:
cmd += ['--net_avg']
if self.do_3D.value:
cmd += ['--do_3D']
cmd += ['--anisotropy', str(anisotropy), '--flow_threshold', str(self.flow_threshold.value), '--cellprob_threshold',
str(self.cellprob_threshold.value), '--stitch_threshold', str(self.stitch_threshold.value), '--min_size', str(self.min_size.value)]
if self.invert.value:
cmd += ['--invert']
if self.remove_edge_masks.value:
cmd += ['--exclude_on_edges']
print(cmd)
try:
subprocess.run(cmd, text=True)
cellpose_output = numpy.load(os.path.join(temp_img_dir, unique_name + "_seg.npy"), allow_pickle=True).item()
y_data = cellpose_output["masks"]
flows = cellpose_output["flows"]
finally:
# Delete the temporary files
try:
shutil.rmtree(temp_dir)
except:
LOGGER.error("Unable to delete temporary directory, files may be in use by another program.")
LOGGER.error("Temp folder is subfolder {tempdir} in your Default Output Folder.\nYou may need to remove it manually.")
y = Objects()
y.segmented = y_data
y.parent_image = x.parent_image
objects = workspace.object_set
objects.add_objects(y, y_name)
if self.save_probabilities.value:
if self.docker_or_python.value == "Docker":
# get rid of extra dimension
prob_map = numpy.squeeze(flows[1], axis=0) # ranges 0-255
else:
prob_map = flows[2]
rescale_prob_map = prob_map.copy()
prob_map01 = numpy.percentile(rescale_prob_map, 1)
prob_map99 = numpy.percentile(rescale_prob_map, 99)
prob_map = numpy.clip((rescale_prob_map - prob_map01) / (prob_map99 - prob_map01), a_min=0, a_max=1)
# Flows come out sized relative to CellPose's inbuilt model size.
# We need to slightly resize to match the original image.
size_corrected = skimage.transform.resize(prob_map, y_data.shape)
prob_image = Image(
size_corrected,
parent_image=x.parent_image,
convert=False,
dimensions=len(size_corrected.shape),
)
workspace.image_set.add(self.probabilities_name.value, prob_image)
if self.show_window:
workspace.display_data.probabilities = size_corrected
self.add_measurements(workspace)
if self.show_window:
if x.volumetric:
# Can't show CellPose-accepted colour images in 3D
workspace.display_data.x_data = x.pixel_data
else:
workspace.display_data.x_data = x_data
workspace.display_data.y_data = y_data
workspace.display_data.dimensions = dimensions
def display(self, workspace, figure):
if self.save_probabilities.value:
layout = (2, 2)
else:
layout = (2, 1)
figure.set_subplots(
dimensions=workspace.display_data.dimensions, subplots=layout
)
figure.subplot_imshow(
colormap="gray",
image=workspace.display_data.x_data,
title="Input Image",
x=0,
y=0,
)
figure.subplot_imshow_labels(
image=workspace.display_data.y_data,
sharexy=figure.subplot(0, 0),
title=self.y_name.value,
x=1,
y=0,
)
if self.save_probabilities.value:
figure.subplot_imshow(
colormap="gray",
image=workspace.display_data.probabilities,
sharexy=figure.subplot(0, 0),
title=self.probabilities_name.value,
x=0,
y=1,
)
def do_check_gpu(self):
import importlib.util
torch_installed = importlib.util.find_spec('torch') is not None
self.cellpose_ver = importlib.metadata.version('cellpose')
#if the old version of cellpose <2.0, then use istorch kwarg
if float(self.cellpose_ver[0:3]) >= 0.7 and int(self.cellpose_ver[0])<2:
GPU_works = core.use_gpu(istorch=torch_installed)
else: # if new version of cellpose, use use_torch kwarg
GPU_works = core.use_gpu(use_torch=torch_installed)
if GPU_works:
message = "GPU appears to be working correctly!"
else:
message = (
"GPU test failed. There may be something wrong with your configuration."
)
import wx
wx.MessageBox(message, caption="GPU Test")
def upgrade_settings(self, setting_values, variable_revision_number, module_name):
if variable_revision_number == 1:
setting_values = setting_values + ["0.4", "0.0"]
variable_revision_number = 2
if variable_revision_number == 2:
setting_values = setting_values + ["0.0", False, "15", "1.0", False, False]
variable_revision_number = 3
if variable_revision_number == 3:
setting_values = [setting_values[0]] + ["Python",CELLPOSE_DOCKER_IMAGE_WITH_PRETRAINED_v232] + setting_values[1:]
variable_revision_number = 4
if variable_revision_number == 4:
setting_values = [setting_values[0]] + ['No'] + setting_values[1:]
variable_revision_number = 5
if variable_revision_number == 5:
setting_values = setting_values + [False]
variable_revision_number = 6
return setting_values, variable_revision_number