tomwer 1.2.9__py3-none-any.whl → 1.3.0a0__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (253) hide show
  1. orangecontrib/tomwer/tutorials/icat_publication.ows +58 -0
  2. orangecontrib/tomwer/widgets/__init__.py +1 -0
  3. orangecontrib/tomwer/widgets/control/DataDiscoveryOW.py +2 -2
  4. orangecontrib/tomwer/widgets/control/DataListOW.py +9 -7
  5. orangecontrib/tomwer/widgets/control/DataSelectorOW.py +21 -10
  6. orangecontrib/tomwer/widgets/control/EDF2NXTomomillOW.py +11 -5
  7. orangecontrib/tomwer/widgets/control/EmailOW.py +4 -4
  8. orangecontrib/tomwer/widgets/control/NXTomomillOW.py +31 -18
  9. orangecontrib/tomwer/widgets/control/NXtomoConcatenate.py +14 -7
  10. orangecontrib/tomwer/widgets/control/NotifierOW.py +1 -0
  11. orangecontrib/tomwer/widgets/control/VolumeSelector.py +7 -4
  12. orangecontrib/tomwer/widgets/control/VolumeSymLinkOW.py +182 -182
  13. orangecontrib/tomwer/widgets/debugtools/DatasetGeneratorOW.py +4 -4
  14. orangecontrib/tomwer/widgets/edit/DarkFlatPatchOW.py +4 -4
  15. orangecontrib/tomwer/widgets/edit/ImageKeyEditorOW.py +3 -3
  16. orangecontrib/tomwer/widgets/edit/ImageKeyUpgraderOW.py +2 -0
  17. orangecontrib/tomwer/widgets/edit/NXtomoEditorOW.py +3 -3
  18. orangecontrib/tomwer/widgets/edit/test/test_nxtomo_editor.py +3 -3
  19. orangecontrib/tomwer/widgets/icat/PublishProcessedDataOW.py +115 -0
  20. orangecontrib/tomwer/widgets/icat/RawDataScreenshotCreatorOW.py +98 -0
  21. orangecontrib/tomwer/widgets/icat/SaveToGalleryAndPublishOW.py +129 -0
  22. orangecontrib/tomwer/widgets/icat/__init__.py +13 -0
  23. orangecontrib/tomwer/widgets/icat/icons/add_gallery.png +0 -0
  24. orangecontrib/tomwer/widgets/icat/icons/add_gallery.svg +82 -0
  25. orangecontrib/tomwer/widgets/icat/icons/publish_processed_data.png +0 -0
  26. orangecontrib/tomwer/widgets/icat/icons/publish_processed_data.svg +95 -0
  27. orangecontrib/tomwer/widgets/icat/icons/raw_screenshots.png +0 -0
  28. orangecontrib/tomwer/widgets/icat/icons/raw_screenshots.svg +143 -0
  29. orangecontrib/tomwer/widgets/icons/tomwer_data_portal.png +0 -0
  30. orangecontrib/tomwer/widgets/icons/tomwer_data_portal.svg +76 -0
  31. orangecontrib/tomwer/widgets/reconstruction/AxisOW.py +9 -8
  32. orangecontrib/tomwer/widgets/reconstruction/CastNabuVolumeOW.py +3 -3
  33. orangecontrib/tomwer/widgets/reconstruction/NabuHelicalPrepareWeightsDoubleOW.py +179 -169
  34. orangecontrib/tomwer/widgets/reconstruction/NabuOW.py +23 -0
  35. orangecontrib/tomwer/widgets/reconstruction/NabuVolumeOW.py +39 -5
  36. orangecontrib/tomwer/widgets/reconstruction/SAAxisOW.py +7 -13
  37. orangecontrib/tomwer/widgets/reconstruction/SADeltaBetaOW.py +7 -17
  38. orangecontrib/tomwer/widgets/reconstruction/SinoNormOW.py +3 -4
  39. orangecontrib/tomwer/widgets/visualization/LivesliceOW.py +1 -1
  40. orangecontrib/tomwer/widgets/visualization/NXtomoMetadataViewerOW.py +3 -3
  41. orangecontrib/tomwer/widgets/visualization/VolumeViewerOW.py +3 -29
  42. tomwer/__main__.py +11 -58
  43. tomwer/app/canvas.py +8 -0
  44. tomwer/app/canvas_launcher/config.py +13 -11
  45. tomwer/app/darkref.py +1 -1
  46. tomwer/app/darkrefpatch.py +1 -1
  47. tomwer/app/imagekeyeditor.py +5 -5
  48. tomwer/app/imagekeyupgrader.py +5 -5
  49. tomwer/app/intensitynormalization.py +2 -2
  50. tomwer/app/radiostack.py +2 -2
  51. tomwer/app/zstitching.py +74 -3
  52. tomwer/core/cluster/cluster.py +26 -0
  53. tomwer/core/log/logger.py +7 -5
  54. tomwer/core/process/conditions/filters.py +1 -1
  55. tomwer/core/process/control/datalistener/datalistener.py +3 -3
  56. tomwer/core/process/control/nxtomoconcatenate.py +13 -13
  57. tomwer/core/process/control/nxtomomill.py +83 -25
  58. tomwer/core/process/control/scantransfer.py +11 -10
  59. tomwer/core/process/control/scanvalidator.py +3 -2
  60. tomwer/core/process/control/test/test_concatenate_nxtomos.py +9 -9
  61. tomwer/core/process/control/test/test_email.py +4 -4
  62. tomwer/core/process/control/test/test_h52nx_process.py +59 -7
  63. tomwer/core/process/control/test/test_volume_link.py +64 -64
  64. tomwer/core/process/control/timer.py +1 -1
  65. tomwer/core/process/control/volumesymlink.py +200 -200
  66. tomwer/core/process/edit/darkflatpatch.py +6 -6
  67. tomwer/core/process/edit/imagekeyeditor.py +17 -18
  68. tomwer/core/process/icat/__init__.py +0 -0
  69. tomwer/core/process/icat/createscreenshots.py +100 -0
  70. tomwer/core/process/icat/gallery.py +377 -0
  71. tomwer/core/process/icat/icatbase.py +36 -0
  72. tomwer/core/process/icat/publish.py +228 -0
  73. tomwer/core/process/icat/screenshots.py +26 -0
  74. tomwer/core/process/output.py +52 -0
  75. tomwer/core/process/reconstruction/axis/axis.py +17 -10
  76. tomwer/core/process/reconstruction/axis/mode.py +4 -0
  77. tomwer/core/process/reconstruction/axis/params.py +9 -4
  78. tomwer/core/process/reconstruction/darkref/darkrefs.py +8 -6
  79. tomwer/core/process/reconstruction/darkref/darkrefscopy.py +1 -1
  80. tomwer/core/process/reconstruction/darkref/params.py +1 -1
  81. tomwer/core/process/reconstruction/lamino/tofu.py +4 -4
  82. tomwer/core/process/reconstruction/nabu/castvolume.py +1 -1
  83. tomwer/core/process/reconstruction/nabu/helical.py +9 -5
  84. tomwer/core/process/reconstruction/nabu/nabucommon.py +32 -62
  85. tomwer/core/process/reconstruction/nabu/nabuscores.py +387 -61
  86. tomwer/core/process/reconstruction/nabu/nabuslices.py +33 -21
  87. tomwer/core/process/reconstruction/nabu/nabuvolume.py +37 -14
  88. tomwer/core/process/reconstruction/nabu/settings.py +2 -2
  89. tomwer/core/process/reconstruction/nabu/utils.py +129 -24
  90. tomwer/core/process/reconstruction/output.py +108 -0
  91. tomwer/core/process/reconstruction/saaxis/saaxis.py +233 -263
  92. tomwer/core/process/reconstruction/sadeltabeta/sadeltabeta.py +140 -86
  93. tomwer/core/process/reconstruction/scores/params.py +4 -1
  94. tomwer/core/process/reconstruction/scores/scores.py +13 -0
  95. tomwer/core/process/reconstruction/test/test_axis_params.py +2 -2
  96. tomwer/core/process/reconstruction/test/test_darkref.py +3 -3
  97. tomwer/core/process/reconstruction/test/test_darkref_copy.py +3 -3
  98. tomwer/core/process/reconstruction/test/test_saaxis.py +3 -4
  99. tomwer/core/process/reconstruction/test/test_sadeltabeta.py +2 -2
  100. tomwer/core/process/stitching/nabustitcher.py +2 -2
  101. tomwer/core/process/test/test_axis.py +6 -6
  102. tomwer/core/process/test/test_dark_and_flat.py +10 -7
  103. tomwer/core/process/test/test_data_transfer.py +7 -6
  104. tomwer/core/process/test/test_nabu.py +4 -4
  105. tomwer/core/process/test/test_normalization.py +2 -2
  106. tomwer/core/scan/edfscan.py +4 -1
  107. tomwer/core/scan/hdf5scan.py +19 -500
  108. tomwer/core/scan/nxtomoscan.py +532 -0
  109. tomwer/core/scan/scanbase.py +42 -20
  110. tomwer/core/scan/scanfactory.py +13 -13
  111. tomwer/core/scan/test/test_future_scan.py +2 -2
  112. tomwer/core/scan/test/test_h5.py +12 -10
  113. tomwer/core/scan/test/test_process_registration.py +2 -2
  114. tomwer/core/scan/test/test_scan.py +4 -3
  115. tomwer/core/settings.py +20 -0
  116. tomwer/core/test/test_scanutils.py +8 -7
  117. tomwer/core/test/test_utils.py +33 -26
  118. tomwer/core/utils/__init__.py +0 -466
  119. tomwer/core/utils/deprecation.py +1 -1
  120. tomwer/core/utils/dictutils.py +14 -0
  121. tomwer/core/utils/lbsram.py +35 -0
  122. tomwer/core/utils/nxtomoutils.py +1 -1
  123. tomwer/core/utils/scanutils.py +6 -6
  124. tomwer/core/utils/spec.py +263 -0
  125. tomwer/core/volume/volumefactory.py +2 -2
  126. tomwer/gui/cluster/slurm.py +260 -60
  127. tomwer/gui/cluster/test/test_cluster.py +13 -0
  128. tomwer/gui/cluster/test/test_supervisor.py +2 -2
  129. tomwer/gui/configuration/__init__.py +0 -0
  130. tomwer/gui/{reconstruction/nabu → configuration}/action.py +1 -32
  131. tomwer/gui/configuration/level.py +22 -0
  132. tomwer/gui/control/actions.py +54 -0
  133. tomwer/gui/control/datalist.py +78 -16
  134. tomwer/gui/control/datalistener.py +4 -16
  135. tomwer/gui/control/{email.py → emailnotifier.py} +9 -18
  136. tomwer/gui/control/history.py +2 -2
  137. tomwer/gui/control/observations.py +2 -2
  138. tomwer/gui/control/reducedarkflatselector.py +1 -1
  139. tomwer/gui/control/selectorwidgetbase.py +36 -9
  140. tomwer/gui/control/serie/seriecreator.py +5 -22
  141. tomwer/gui/control/test/test_email.py +1 -1
  142. tomwer/gui/control/test/test_scanvalidator.py +6 -5
  143. tomwer/gui/control/test/test_single_tomo_obj.py +2 -2
  144. tomwer/gui/control/tomoobjdisplaymode.py +8 -0
  145. tomwer/gui/debugtools/datasetgenerator.py +3 -3
  146. tomwer/gui/edit/dkrfpatch.py +16 -22
  147. tomwer/gui/edit/imagekeyeditor.py +8 -11
  148. tomwer/gui/edit/nxtomoeditor.py +111 -44
  149. tomwer/gui/edit/nxtomowarmer.py +4 -4
  150. tomwer/gui/edit/test/test_dkrf_patch.py +7 -7
  151. tomwer/gui/edit/test/test_image_key_editor.py +3 -3
  152. tomwer/gui/edit/test/test_nx_editor.py +40 -16
  153. tomwer/gui/icat/__init__.py +0 -0
  154. tomwer/gui/icat/createscreenshots.py +80 -0
  155. tomwer/gui/icat/gallery.py +214 -0
  156. tomwer/gui/icat/publish.py +187 -0
  157. tomwer/gui/reconstruction/axis/axis.py +171 -57
  158. tomwer/gui/reconstruction/axis/radioaxis.py +80 -95
  159. tomwer/gui/reconstruction/darkref/darkrefcopywidget.py +3 -2
  160. tomwer/gui/reconstruction/lamino/tofu/projections.py +1 -1
  161. tomwer/gui/reconstruction/lamino/tofu/tofuoutput.py +3 -6
  162. tomwer/gui/reconstruction/nabu/castvolume.py +1 -1
  163. tomwer/gui/reconstruction/nabu/check.py +9 -9
  164. tomwer/gui/reconstruction/nabu/helical.py +29 -12
  165. tomwer/gui/reconstruction/nabu/nabuconfig/base.py +2 -4
  166. tomwer/gui/reconstruction/nabu/nabuconfig/output.py +110 -33
  167. tomwer/gui/reconstruction/nabu/nabuconfig/phase.py +9 -12
  168. tomwer/gui/reconstruction/nabu/nabuconfig/preprocessing.py +219 -29
  169. tomwer/gui/reconstruction/nabu/nabuconfig/reconstruction.py +3 -6
  170. tomwer/gui/reconstruction/nabu/nabuflow.py +12 -20
  171. tomwer/gui/reconstruction/nabu/slices.py +6 -7
  172. tomwer/gui/reconstruction/nabu/volume.py +22 -10
  173. tomwer/gui/reconstruction/normalization/intensity.py +15 -23
  174. tomwer/gui/reconstruction/saaxis/corrangeselector.py +7 -23
  175. tomwer/gui/reconstruction/saaxis/dimensionwidget.py +1 -1
  176. tomwer/gui/reconstruction/saaxis/saaxis.py +7 -9
  177. tomwer/gui/reconstruction/sadeltabeta/saadeltabeta.py +2 -1
  178. tomwer/gui/reconstruction/scores/control.py +2 -9
  179. tomwer/gui/reconstruction/scores/scoreplot.py +11 -5
  180. tomwer/gui/reconstruction/test/test_axis.py +23 -12
  181. tomwer/gui/reconstruction/test/test_lamino.py +8 -3
  182. tomwer/gui/reconstruction/test/test_nabu.py +28 -9
  183. tomwer/gui/reconstruction/test/test_saaxis.py +3 -3
  184. tomwer/gui/reconstruction/test/test_sadeltabeta.py +2 -2
  185. tomwer/gui/settings.py +5 -28
  186. tomwer/gui/stackplot.py +2 -5
  187. tomwer/gui/stitching/action.py +49 -0
  188. tomwer/gui/stitching/config/axisparams.py +7 -24
  189. tomwer/gui/stitching/config/output.py +10 -8
  190. tomwer/gui/stitching/config/positionoveraxis.py +22 -23
  191. tomwer/gui/stitching/normalization.py +117 -0
  192. tomwer/gui/stitching/stitchandbackground.py +4 -6
  193. tomwer/gui/stitching/stitching.py +265 -43
  194. tomwer/gui/stitching/stitching_preview.py +62 -5
  195. tomwer/gui/stitching/stitching_raw.py +2 -5
  196. tomwer/gui/stitching/z_stitching/fineestimation.py +0 -60
  197. tomwer/gui/utils/buttons.py +112 -29
  198. tomwer/gui/utils/inputwidget.py +33 -25
  199. tomwer/gui/utils/scandescription.py +4 -0
  200. tomwer/gui/utils/step.py +144 -0
  201. tomwer/gui/utils/unitsystem.py +2 -5
  202. tomwer/gui/utils/vignettes.py +176 -15
  203. tomwer/gui/visualization/dataviewer.py +1 -4
  204. tomwer/gui/visualization/diffviewer/diffviewer.py +7 -16
  205. tomwer/gui/visualization/diffviewer/shiftwidget.py +2 -5
  206. tomwer/gui/visualization/scanoverview.py +1 -1
  207. tomwer/gui/visualization/sinogramviewer.py +1 -10
  208. tomwer/gui/visualization/test/test_diffviewer.py +3 -3
  209. tomwer/gui/visualization/test/test_nx_tomo_metadata_viewer.py +4 -4
  210. tomwer/gui/visualization/test/test_sinogramviewer.py +2 -2
  211. tomwer/gui/visualization/test/test_stacks.py +3 -3
  212. tomwer/gui/visualization/test/test_volumeviewer.py +2 -2
  213. tomwer/io/utils/raw_and_processed_data.py +84 -0
  214. tomwer/io/utils/tomoobj.py +4 -6
  215. tomwer/resources/gui/icons/ruler.png +0 -0
  216. tomwer/resources/gui/icons/ruler.svg +273 -0
  217. tomwer/resources/gui/icons/short_description.png +0 -0
  218. tomwer/resources/gui/icons/short_description.svg +58 -0
  219. tomwer/resources/gui/icons/url.png +0 -0
  220. tomwer/resources/gui/icons/url.svg +58 -0
  221. tomwer/synctools/stacks/edit/darkflatpatch.py +2 -2
  222. tomwer/synctools/stacks/edit/imagekeyeditor.py +2 -2
  223. tomwer/synctools/stacks/reconstruction/axis.py +4 -4
  224. tomwer/synctools/stacks/reconstruction/castvolume.py +2 -2
  225. tomwer/synctools/stacks/reconstruction/dkrefcopy.py +4 -10
  226. tomwer/synctools/stacks/reconstruction/nabu.py +2 -2
  227. tomwer/synctools/stacks/reconstruction/normalization.py +2 -2
  228. tomwer/synctools/stacks/reconstruction/saaxis.py +2 -2
  229. tomwer/synctools/stacks/reconstruction/sadeltabeta.py +2 -2
  230. tomwer/synctools/test/test_darkRefs.py +7 -58
  231. tomwer/synctools/test/test_foldertransfer.py +6 -6
  232. tomwer/synctools/utils/scanstages.py +6 -6
  233. tomwer/tests/conftest.py +34 -0
  234. tomwer/tests/datasets.py +13 -0
  235. tomwer/tests/test_scripts.py +92 -39
  236. tomwer/tests/utils.py +5 -0
  237. tomwer/version.py +3 -3
  238. {tomwer-1.2.9.dist-info → tomwer-1.3.0a0.dist-info}/METADATA +39 -39
  239. {tomwer-1.2.9.dist-info → tomwer-1.3.0a0.dist-info}/RECORD +248 -209
  240. tomwer/resources/gui/icons/esrf_1.svg +0 -307
  241. tomwer/resources/gui/icons/triangle.svg +0 -80
  242. tomwer/synctools/test/test_scanstages.py +0 -162
  243. tomwer/tests/utils/__init__.py +0 -247
  244. tomwer/tests/utils/utilstest.py +0 -220
  245. /tomwer/app/{saaxis.py → multicor.py} +0 -0
  246. /tomwer/app/{sadeltabeta.py → multipag.py} +0 -0
  247. /tomwer/core/process/control/{email.py → emailnotifier.py} +0 -0
  248. /tomwer-1.2.9-py3.11-nspkg.pth → /tomwer-1.3.0a0-py3.11-nspkg.pth +0 -0
  249. {tomwer-1.2.9.dist-info → tomwer-1.3.0a0.dist-info}/LICENSE +0 -0
  250. {tomwer-1.2.9.dist-info → tomwer-1.3.0a0.dist-info}/WHEEL +0 -0
  251. {tomwer-1.2.9.dist-info → tomwer-1.3.0a0.dist-info}/entry_points.txt +0 -0
  252. {tomwer-1.2.9.dist-info → tomwer-1.3.0a0.dist-info}/namespace_packages.txt +0 -0
  253. {tomwer-1.2.9.dist-info → tomwer-1.3.0a0.dist-info}/top_level.txt +0 -0
@@ -1,200 +1,200 @@
1
- # coding: utf-8
2
- # /*##########################################################################
3
- #
4
- # Copyright (c) 2016-2017 European Synchrotron Radiation Facility
5
- #
6
- # Permission is hereby granted, free of charge, to any person obtaining a copy
7
- # of this software and associated documentation files (the "Software"), to deal
8
- # in the Software without restriction, including without limitation the rights
9
- # to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
10
- # copies of the Software, and to permit persons to whom the Software is
11
- # furnished to do so, subject to the following conditions:
12
- #
13
- # The above copyright notice and this permission notice shall be included in
14
- # all copies or substantial portions of the Software.
15
- #
16
- # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17
- # IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18
- # FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
19
- # AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20
- # LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
21
- # OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
22
- # THE SOFTWARE.
23
- #
24
- # ###########################################################################*/
25
-
26
- __authors__ = ["H. Payno"]
27
- __license__ = "MIT"
28
- __date__ = "16/11/2020"
29
-
30
-
31
- import logging
32
- import os
33
-
34
- from silx.utils.enum import Enum as _Enum
35
- from silx.utils.deprecation import deprecated_warning, deprecated
36
- from tomoscan.factory import Factory
37
- from tomoscan.volumebase import VolumeBase
38
-
39
- import tomwer.version
40
- from tomwer.core.process.task import Task
41
- from tomwer.core.scan.scanbase import TomwerScanBase
42
- from tomwer.core.scan.scanfactory import ScanFactory
43
- from tomwer.core.utils.scanutils import data_identifier_to_scan
44
- from tomwer.utils import docstring
45
-
46
- _logger = logging.getLogger(__name__)
47
-
48
-
49
- class OutputType(_Enum):
50
- STATIC = "static"
51
- ONE_LEVEL_UPPER = "../volume"
52
-
53
-
54
- def create_volume_symbolic_link(scan: TomwerScanBase, output_folder: str):
55
- """
56
- Create a symbolic link for each volume reconstructed of `scan`
57
-
58
- :param TomwerScanBase scan:
59
- :param str output_folder:
60
- """
61
- if scan.latest_vol_reconstructions is None:
62
- _logger.info("No volume reconstructed found")
63
- return
64
- if not os.path.exists(output_folder):
65
- os.makedirs(output_folder)
66
- for volume_url in scan.latest_vol_reconstructions:
67
- volume = Factory.create_tomo_object_from_identifier(volume_url)
68
- assert isinstance(volume, VolumeBase)
69
- volume_files = set(
70
- list(volume.browse_data_files()) + list(volume.browse_metadata_files()),
71
- )
72
- for src_file_path in volume_files:
73
- if src_file_path is not None:
74
- dst_file_path = os.path.join(
75
- output_folder, os.path.basename(src_file_path)
76
- )
77
- if os.path.exists(dst_file_path):
78
- _logger.info(
79
- "{} already exists. Cannot create a symbolic link on it"
80
- )
81
- else:
82
- print(
83
- "src_file is", src_file_path, "dst_file_path is", dst_file_path
84
- )
85
- os.symlink(src=src_file_path, dst=dst_file_path)
86
-
87
-
88
- class VolumeSymbolicLinkTask(
89
- Task,
90
- input_names=("data",),
91
- output_names=("data",),
92
- optional_input_names=("serialize_output_data", "output_type", "output_folder"),
93
- ):
94
- """
95
- Process class for volume symbolic link
96
- """
97
-
98
- def __init__(
99
- self, varinfo=None, inputs=None, node_id=None, node_attrs=None, execinfo=None
100
- ):
101
- super().__init__(
102
- varinfo=varinfo,
103
- inputs=inputs,
104
- node_id=node_id,
105
- node_attrs=node_attrs,
106
- execinfo=execinfo,
107
- )
108
- self._output_type = OutputType.ONE_LEVEL_UPPER
109
- self._output_folder = None
110
-
111
- @docstring(Task.set_configuration)
112
- @deprecated(
113
- reason="ewoksification",
114
- replacement="task.inputs.output_type and task.inputs.output_folder",
115
- since_version="1.2",
116
- )
117
- def set_configuration(self, properties):
118
- if "output_type" in properties:
119
- self._output_type = OutputType.from_value(properties["output_type"])
120
- if "output_folder" in properties:
121
- self._output_folder = properties["output_folder"]
122
-
123
- @deprecated(
124
- reason="ewoksification",
125
- replacement="task.inputs.output_type and task.inputs.output_folder",
126
- since_version="1.2",
127
- )
128
- def get_configuration(self):
129
- return super().get_configuration()
130
-
131
- @docstring(Task.program_name)
132
- @staticmethod
133
- def program_name():
134
- return "tomwer_volume_symlink"
135
-
136
- @docstring(Task.program_version)
137
- @staticmethod
138
- def program_version():
139
- return tomwer.version.version
140
-
141
- @docstring(Task.run)
142
- def run(self):
143
- scan = data_identifier_to_scan(self.inputs.data)
144
- if type(scan) is dict:
145
- scan = ScanFactory.create_scan_object_frm_dict(scan)
146
- else:
147
- scan = scan
148
- if scan is None:
149
- return
150
- if not isinstance(scan, TomwerScanBase):
151
- raise TypeError(
152
- f"scan is expected to be a dict or an instance of TomwerScanBase. Not {type(scan)}"
153
- )
154
-
155
- output_type = OutputType.from_value(
156
- self.get_input_value("output_type", self._output_type)
157
- )
158
- output_folder = self.get_input_value("output_folder", self._output_folder)
159
- if output_type is OutputType.STATIC:
160
- if output_folder is None:
161
- raise ValueError(
162
- "Manual setting of the output folder is "
163
- "requested but None is provided."
164
- )
165
- else:
166
- if not os.path.isabs(output_folder):
167
- os.path.abspath(
168
- os.path.join(os.path.realpath(scan.path), output_folder)
169
- )
170
- elif output_type is OutputType.ONE_LEVEL_UPPER:
171
- output_folder = os.path.realpath(scan.path)
172
- output_folder = os.path.abspath(
173
- os.path.join(output_folder, output_type.value)
174
- )
175
- else:
176
- raise ValueError(f"output type {output_type.value} is not managed")
177
- create_volume_symbolic_link(scan=scan, output_folder=output_folder)
178
- if self.get_input_value("serialize_output_data", True):
179
- self.outputs.data = scan.to_dict()
180
- else:
181
- self.outputs.data = scan
182
-
183
- @docstring(Task.definition)
184
- @staticmethod
185
- def definition():
186
- return "Create a symbolic link to the volume folder"
187
-
188
-
189
- class VolumeSymbolicLinkProcess(VolumeSymbolicLinkTask):
190
- def __init__(
191
- self, varinfo=None, inputs=None, node_id=None, node_attrs=None, execinfo=None
192
- ):
193
- deprecated_warning(
194
- name="tomwer.core.process.control.volumesymlink.VolumeSymbolicLinkProcess",
195
- type_="class",
196
- reason="improve readibility",
197
- since_version="1.2",
198
- replacement="VolumeSymbolicLinkTask",
199
- )
200
- super().__init__(varinfo, inputs, node_id, node_attrs, execinfo)
1
+ # # coding: utf-8
2
+ # # /*##########################################################################
3
+ # #
4
+ # # Copyright (c) 2016-2017 European Synchrotron Radiation Facility
5
+ # #
6
+ # # Permission is hereby granted, free of charge, to any person obtaining a copy
7
+ # # of this software and associated documentation files (the "Software"), to deal
8
+ # # in the Software without restriction, including without limitation the rights
9
+ # # to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
10
+ # # copies of the Software, and to permit persons to whom the Software is
11
+ # # furnished to do so, subject to the following conditions:
12
+ # #
13
+ # # The above copyright notice and this permission notice shall be included in
14
+ # # all copies or substantial portions of the Software.
15
+ # #
16
+ # # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17
+ # # IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18
+ # # FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
19
+ # # AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20
+ # # LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
21
+ # # OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
22
+ # # THE SOFTWARE.
23
+ # #
24
+ # # ###########################################################################*/
25
+
26
+ # __authors__ = ["H. Payno"]
27
+ # __license__ = "MIT"
28
+ # __date__ = "16/11/2020"
29
+
30
+
31
+ # import logging
32
+ # import os
33
+
34
+ # from silx.utils.enum import Enum as _Enum
35
+ # from tomwer.core.utils.deprecation import deprecated_warning, deprecated
36
+ # from tomoscan.factory import Factory
37
+ # from tomoscan.volumebase import VolumeBase
38
+
39
+ # import tomwer.version
40
+ # from tomwer.core.process.task import Task
41
+ # from tomwer.core.scan.scanbase import TomwerScanBase
42
+ # from tomwer.core.scan.scanfactory import ScanFactory
43
+ # from tomwer.core.utils.scanutils import data_identifier_to_scan
44
+ # from tomwer.utils import docstring
45
+
46
+ # _logger = logging.getLogger(__name__)
47
+
48
+
49
+ # class OutputType(_Enum):
50
+ # STATIC = "static"
51
+ # ONE_LEVEL_UPPER = "../volume"
52
+
53
+
54
+ # def create_volume_symbolic_link(scan: TomwerScanBase, output_folder: str):
55
+ # """
56
+ # Create a symbolic link for each volume reconstructed of `scan`
57
+
58
+ # :param TomwerScanBase scan:
59
+ # :param str output_folder:
60
+ # """
61
+ # if scan.latest_vol_reconstructions is None:
62
+ # _logger.info("No volume reconstructed found")
63
+ # return
64
+ # if not os.path.exists(output_folder):
65
+ # os.makedirs(output_folder)
66
+ # for volume_url in scan.latest_vol_reconstructions:
67
+ # volume = Factory.create_tomo_object_from_identifier(volume_url)
68
+ # assert isinstance(volume, VolumeBase)
69
+ # volume_files = set(
70
+ # list(volume.browse_data_files()) + list(volume.browse_metadata_files()),
71
+ # )
72
+ # for src_file_path in volume_files:
73
+ # if src_file_path is not None:
74
+ # dst_file_path = os.path.join(
75
+ # output_folder, os.path.basename(src_file_path)
76
+ # )
77
+ # if os.path.exists(dst_file_path):
78
+ # _logger.info(
79
+ # "{} already exists. Cannot create a symbolic link on it"
80
+ # )
81
+ # else:
82
+ # print(
83
+ # "src_file is", src_file_path, "dst_file_path is", dst_file_path
84
+ # )
85
+ # os.symlink(src=src_file_path, dst=dst_file_path)
86
+
87
+
88
+ # class VolumeSymbolicLinkTask(
89
+ # Task,
90
+ # input_names=("data",),
91
+ # output_names=("data",),
92
+ # optional_input_names=("serialize_output_data", "output_type", "output_folder"),
93
+ # ):
94
+ # """
95
+ # Process class for volume symbolic link
96
+ # """
97
+
98
+ # def __init__(
99
+ # self, varinfo=None, inputs=None, node_id=None, node_attrs=None, execinfo=None
100
+ # ):
101
+ # super().__init__(
102
+ # varinfo=varinfo,
103
+ # inputs=inputs,
104
+ # node_id=node_id,
105
+ # node_attrs=node_attrs,
106
+ # execinfo=execinfo,
107
+ # )
108
+ # self._output_type = OutputType.ONE_LEVEL_UPPER
109
+ # self._output_folder = None
110
+
111
+ # @docstring(Task.set_configuration)
112
+ # @deprecated(
113
+ # reason="ewoksification",
114
+ # replacement="task.inputs.output_type and task.inputs.output_folder",
115
+ # since_version="1.2",
116
+ # )
117
+ # def set_configuration(self, properties):
118
+ # if "output_type" in properties:
119
+ # self._output_type = OutputType.from_value(properties["output_type"])
120
+ # if "output_folder" in properties:
121
+ # self._output_folder = properties["output_folder"]
122
+
123
+ # @deprecated(
124
+ # reason="ewoksification",
125
+ # replacement="task.inputs.output_type and task.inputs.output_folder",
126
+ # since_version="1.2",
127
+ # )
128
+ # def get_configuration(self):
129
+ # return super().get_configuration()
130
+
131
+ # @docstring(Task.program_name)
132
+ # @staticmethod
133
+ # def program_name():
134
+ # return "tomwer_volume_symlink"
135
+
136
+ # @docstring(Task.program_version)
137
+ # @staticmethod
138
+ # def program_version():
139
+ # return tomwer.version.version
140
+
141
+ # @docstring(Task.run)
142
+ # def run(self):
143
+ # scan = data_identifier_to_scan(self.inputs.data)
144
+ # if type(scan) is dict:
145
+ # scan = ScanFactory.create_scan_object_frm_dict(scan)
146
+ # else:
147
+ # scan = scan
148
+ # if scan is None:
149
+ # return
150
+ # if not isinstance(scan, TomwerScanBase):
151
+ # raise TypeError(
152
+ # f"scan is expected to be a dict or an instance of TomwerScanBase. Not {type(scan)}"
153
+ # )
154
+
155
+ # output_type = OutputType.from_value(
156
+ # self.get_input_value("output_type", self._output_type)
157
+ # )
158
+ # output_folder = self.get_input_value("output_folder", self._output_folder)
159
+ # if output_type is OutputType.STATIC:
160
+ # if output_folder is None:
161
+ # raise ValueError(
162
+ # "Manual setting of the output folder is "
163
+ # "requested but None is provided."
164
+ # )
165
+ # else:
166
+ # if not os.path.isabs(output_folder):
167
+ # os.path.abspath(
168
+ # os.path.join(os.path.realpath(scan.path), output_folder)
169
+ # )
170
+ # elif output_type is OutputType.ONE_LEVEL_UPPER:
171
+ # output_folder = os.path.realpath(scan.path)
172
+ # output_folder = os.path.abspath(
173
+ # os.path.join(output_folder, output_type.value)
174
+ # )
175
+ # else:
176
+ # raise ValueError(f"output type {output_type.value} is not managed")
177
+ # create_volume_symbolic_link(scan=scan, output_folder=output_folder)
178
+ # if self.get_input_value("serialize_output_data", True):
179
+ # self.outputs.data = scan.to_dict()
180
+ # else:
181
+ # self.outputs.data = scan
182
+
183
+ # @docstring(Task.definition)
184
+ # @staticmethod
185
+ # def definition():
186
+ # return "Create a symbolic link to the volume folder"
187
+
188
+
189
+ # class VolumeSymbolicLinkProcess(VolumeSymbolicLinkTask):
190
+ # def __init__(
191
+ # self, varinfo=None, inputs=None, node_id=None, node_attrs=None, execinfo=None
192
+ # ):
193
+ # deprecated_warning(
194
+ # name="tomwer.core.process.control.volumesymlink.VolumeSymbolicLinkProcess",
195
+ # type_="class",
196
+ # reason="improve readibility",
197
+ # since_version="1.2",
198
+ # replacement="VolumeSymbolicLinkTask",
199
+ # )
200
+ # super().__init__(varinfo, inputs, node_id, node_attrs, execinfo)
@@ -31,24 +31,24 @@ __date__ = "04/11/2020"
31
31
  import nxtomomill.version
32
32
  from nxtomomill.utils import add_dark_flat_nx_file
33
33
  from silx.io.url import DataUrl
34
- from silx.utils.deprecation import deprecated_warning, deprecated
34
+ from tomwer.core.utils.deprecation import deprecated_warning, deprecated
35
35
 
36
36
  from tomwer.core.process.task import Task
37
- from tomwer.core.scan.hdf5scan import HDF5TomoScan
37
+ from tomwer.core.scan.nxtomoscan import NXtomoScan
38
38
  from tomwer.core.scan.scanbase import TomwerScanBase
39
39
  from tomwer.core.utils.scanutils import data_identifier_to_scan
40
40
 
41
41
 
42
- def apply_dark_flat_patch(scan: HDF5TomoScan, config: dict) -> TomwerScanBase:
42
+ def apply_dark_flat_patch(scan: NXtomoScan, config: dict) -> TomwerScanBase:
43
43
  """
44
44
 
45
45
  :param scan:
46
46
  :param config:
47
47
  :return:
48
48
  """
49
- if not isinstance(scan, HDF5TomoScan):
49
+ if not isinstance(scan, NXtomoScan):
50
50
  raise ValueError(
51
- f"Dark and flat patch only manage HDF5TomoScan and not {type(scan)}"
51
+ f"Dark and flat patch only manage NXtomoScan and not {type(scan)}"
52
52
  )
53
53
  if config is None:
54
54
  return scan
@@ -82,7 +82,7 @@ class DarkFlatPatchTask(
82
82
  raise TypeError(
83
83
  f"scan is expected to be a dict or an instance of TomwerScanBase. Not {type(scan)}"
84
84
  )
85
- if not isinstance(scan, HDF5TomoScan):
85
+ if not isinstance(scan, NXtomoScan):
86
86
  raise ValueError(f"input type of {scan}: {type(scan)} is not managed")
87
87
 
88
88
  config = self.inputs.configuration
@@ -32,12 +32,11 @@ import logging
32
32
 
33
33
  import nxtomomill.version
34
34
  from nxtomomill.utils import change_image_key_control as _change_image_key_control
35
- from tomoscan.esrf.scan.hdf5scan import ImageKey
36
- from tomoscan.esrf.scan.hdf5scan import ImageKey as _ImageKey
37
- from silx.utils.deprecation import deprecated_warning, deprecated
35
+ from nxtomo.nxobject.nxdetector import ImageKey
36
+ from tomwer.core.utils.deprecation import deprecated_warning, deprecated
38
37
 
39
38
  from tomwer.core.process.task import Task
40
- from tomwer.core.scan.hdf5scan import HDF5TomoScan
39
+ from tomwer.core.scan.nxtomoscan import NXtomoScan
41
40
  from tomwer.core.scan.scanbase import TomwerScanBase
42
41
  from tomwer.core.utils.scanutils import data_identifier_to_scan
43
42
 
@@ -45,14 +44,14 @@ _logger = logging.getLogger(__name__)
45
44
 
46
45
 
47
46
  IMAGE_KEYS = {
48
- "projection": _ImageKey.PROJECTION,
49
- "invalid": _ImageKey.INVALID,
50
- "dark": _ImageKey.DARK_FIELD,
51
- "flat": _ImageKey.FLAT_FIELD,
47
+ "projection": ImageKey.PROJECTION,
48
+ "invalid": ImageKey.INVALID,
49
+ "dark": ImageKey.DARK_FIELD,
50
+ "flat": ImageKey.FLAT_FIELD,
52
51
  }
53
52
 
54
53
 
55
- def change_image_key_control(scan: HDF5TomoScan, config: dict) -> TomwerScanBase:
54
+ def change_image_key_control(scan: NXtomoScan, config: dict) -> TomwerScanBase:
56
55
  """
57
56
 
58
57
  :param scan:
@@ -63,9 +62,9 @@ def change_image_key_control(scan: HDF5TomoScan, config: dict) -> TomwerScanBase
63
62
  """
64
63
  if scan is None:
65
64
  return
66
- elif not isinstance(scan, HDF5TomoScan):
65
+ elif not isinstance(scan, NXtomoScan):
67
66
  raise ValueError(
68
- f"Image key control only handle HDF5TomoScan and not {type(scan)}"
67
+ f"Image key control only handle NXtomoScan and not {type(scan)}"
69
68
  )
70
69
 
71
70
  if "modifications" not in config:
@@ -125,7 +124,7 @@ class ImageKeyEditorTask(
125
124
  raise TypeError(
126
125
  f"scan is expected to be a dict or an instance of TomwerScanBase. Not {type(scan)}"
127
126
  )
128
- if not isinstance(scan, HDF5TomoScan):
127
+ if not isinstance(scan, NXtomoScan):
129
128
  raise ValueError(f"input type of {scan}: {(type(scan))} is not managed")
130
129
 
131
130
  config = self.inputs.configuration
@@ -178,8 +177,8 @@ class ImageKeyUpgraderTask(
178
177
 
179
178
  def run(self):
180
179
  scan = data_identifier_to_scan(self.inputs.data)
181
- if not isinstance(scan, HDF5TomoScan):
182
- raise TypeError(f"scan is expected to be an instance of {HDF5TomoScan}")
180
+ if not isinstance(scan, NXtomoScan):
181
+ raise TypeError(f"scan is expected to be an instance of {NXtomoScan}")
183
182
  operations = self.inputs.operations
184
183
  if not isinstance(operations, dict):
185
184
  raise TypeError("operations is expected to be an dict")
@@ -204,7 +203,7 @@ class ImageKeyUpgraderTask(
204
203
  process_file=scan.process_file,
205
204
  entry=scan.entry,
206
205
  configuration={
207
- _ImageKey.from_value(key).value: _ImageKey.from_value(value).value
206
+ ImageKey.from_value(key).value: ImageKey.from_value(value).value
208
207
  for key, value in operations.items()
209
208
  },
210
209
  results={},
@@ -218,13 +217,13 @@ class ImageKeyUpgraderTask(
218
217
 
219
218
  @staticmethod
220
219
  def from_operation_to_config(
221
- scan: TomwerScanBase, from_image_key: _ImageKey, to_image_key: _ImageKey
220
+ scan: TomwerScanBase, from_image_key: ImageKey, to_image_key: ImageKey
222
221
  ):
223
222
  """
224
223
  retrieve frame indices to be updated
225
224
  """
226
- from_image_key = _ImageKey.from_value(from_image_key)
227
- to_image_key = _ImageKey.from_value(to_image_key)
225
+ from_image_key = ImageKey.from_value(from_image_key)
226
+ to_image_key = ImageKey.from_value(to_image_key)
228
227
 
229
228
  config = {}
230
229
  for i_frame, frame in enumerate(scan.frames):
File without changes
@@ -0,0 +1,100 @@
1
+ import numpy
2
+ from tomwer.core.process.task import Task
3
+ from tomwer.core.scan.scanbase import TomwerScanBase
4
+ from tomwer.core.utils.dictutils import concatenate_dict
5
+ from tomwer.core.process.icat.screenshots import IcatScreenshots
6
+ from tomwer.core.process.icat.gallery import deduce_dataset_gallery_location
7
+
8
+ from processview.core.manager import DatasetState, ProcessManager
9
+
10
+
11
+ def get_closest_projection(angle, angles_list):
12
+ idx_closest = numpy.argmin(numpy.abs(angles_list - angle))
13
+ return angles_list[idx_closest]
14
+
15
+
16
+ def select_angles(angles_list: tuple, each_angle: int) -> tuple:
17
+ angles_list = sorted(angles_list)
18
+ if len(angles_list) > 0:
19
+ start_angle = angles_list[0]
20
+ stop_angle = angles_list[-1]
21
+ picked_angle = numpy.arange(start_angle, stop_angle + 1, step=each_angle)
22
+ return tuple(
23
+ [get_closest_projection(angle, angles_list) for angle in picked_angle]
24
+ )
25
+ else:
26
+ return tuple()
27
+
28
+
29
+ class CreateRawDataScreenshotsTask(
30
+ Task,
31
+ input_names=("data",), # screenshots as instance of :class:Screenshots
32
+ optional_input_names=(
33
+ "__process__",
34
+ "raw_projections_required",
35
+ "raw_projections_each",
36
+ "raw_darks_required",
37
+ "raw_flats_required",
38
+ ),
39
+ output_names=("screenshots",),
40
+ ):
41
+ """
42
+ simple task to create screenshots from raw data
43
+ One raw projection will be picked each 'raw_projections_each' angle (expected in degree)
44
+ If 'with_flat' then will pick the first flat of each serie
45
+ If 'with_dark' then will pick the first dark of each serie
46
+ """
47
+
48
+ def run(self):
49
+ process = self.get_input_value("__process__", None)
50
+ scan = self.inputs.data
51
+ if not isinstance(scan, TomwerScanBase):
52
+ raise TypeError(
53
+ f"scan is expected to be an instance of {TomwerScanBase}. Get {type(scan)} instead"
54
+ )
55
+ if process is not None:
56
+ ProcessManager().notify_dataset_state(
57
+ dataset=scan,
58
+ process=process(),
59
+ state=DatasetState.ON_GOING,
60
+ )
61
+
62
+ raw_projections_required = self.get_input_value(
63
+ "raw_projections_required", True
64
+ )
65
+ proj_each = self.get_input_value("raw_projections_each", 90)
66
+ raw_flats_required = self.get_input_value("raw_flats_required", True)
67
+ raw_darks_required = self.get_input_value("raw_darks_required", True)
68
+
69
+ screenshots_urls = {}
70
+ # dict with screenshot name as key and DataUrl as value
71
+ if raw_darks_required and len(scan.darks) > 0:
72
+ screenshots_urls["dark"] = next(iter(scan.darks.values()))
73
+ if raw_flats_required and len(scan.flats) > 0:
74
+ screenshots_urls["flat"] = next(iter(scan.flats.values()))
75
+ if raw_projections_required:
76
+ projections = scan.projections_with_angle()
77
+ picked_angles = select_angles(
78
+ angles_list=sorted(projections.keys()),
79
+ each_angle=proj_each,
80
+ )
81
+ screenshots_urls = concatenate_dict(
82
+ screenshots_urls,
83
+ {
84
+ f"projection_{angle:.1f}": projections[angle]
85
+ for angle in picked_angles
86
+ },
87
+ )
88
+
89
+ self.outputs.screenshots = IcatScreenshots(
90
+ data_dir=deduce_dataset_gallery_location(scan),
91
+ screenshots=screenshots_urls,
92
+ scan=scan,
93
+ )
94
+
95
+ if process is not None:
96
+ ProcessManager().notify_dataset_state(
97
+ dataset=scan,
98
+ process=process(),
99
+ state=DatasetState.SUCCEED,
100
+ )