dls-dodal 1.48.0__py3-none-any.whl → 1.50.0__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 (41) hide show
  1. {dls_dodal-1.48.0.dist-info → dls_dodal-1.50.0.dist-info}/METADATA +2 -1
  2. {dls_dodal-1.48.0.dist-info → dls_dodal-1.50.0.dist-info}/RECORD +41 -30
  3. dodal/_version.py +2 -2
  4. dodal/beamlines/aithre.py +15 -0
  5. dodal/beamlines/b16.py +65 -0
  6. dodal/beamlines/b18.py +38 -0
  7. dodal/beamlines/i10.py +41 -233
  8. dodal/beamlines/k11.py +35 -0
  9. dodal/common/beamlines/device_helpers.py +1 -0
  10. dodal/devices/apple2_undulator.py +257 -136
  11. dodal/devices/b16/__init__.py +0 -0
  12. dodal/devices/b16/detector.py +34 -0
  13. dodal/devices/bimorph_mirror.py +29 -36
  14. dodal/devices/electron_analyser/__init__.py +12 -2
  15. dodal/devices/electron_analyser/abstract/base_detector.py +3 -128
  16. dodal/devices/electron_analyser/abstract/base_driver_io.py +8 -3
  17. dodal/devices/electron_analyser/abstract/base_region.py +6 -3
  18. dodal/devices/electron_analyser/detector.py +141 -0
  19. dodal/devices/electron_analyser/enums.py +6 -0
  20. dodal/devices/electron_analyser/specs/__init__.py +2 -0
  21. dodal/devices/electron_analyser/specs/detector.py +1 -1
  22. dodal/devices/electron_analyser/specs/driver_io.py +4 -5
  23. dodal/devices/electron_analyser/specs/enums.py +8 -0
  24. dodal/devices/electron_analyser/specs/region.py +3 -2
  25. dodal/devices/electron_analyser/types.py +30 -4
  26. dodal/devices/electron_analyser/util.py +1 -1
  27. dodal/devices/electron_analyser/vgscienta/__init__.py +2 -0
  28. dodal/devices/electron_analyser/vgscienta/detector.py +1 -1
  29. dodal/devices/electron_analyser/vgscienta/driver_io.py +2 -1
  30. dodal/devices/electron_analyser/vgscienta/enums.py +19 -0
  31. dodal/devices/electron_analyser/vgscienta/region.py +7 -22
  32. dodal/devices/hutch_shutter.py +6 -6
  33. dodal/devices/i10/__init__.py +0 -0
  34. dodal/devices/i10/i10_apple2.py +181 -126
  35. dodal/devices/i22/nxsas.py +1 -1
  36. dodal/devices/oav/oav_detector.py +45 -7
  37. dodal/plans/bimorph.py +333 -0
  38. {dls_dodal-1.48.0.dist-info → dls_dodal-1.50.0.dist-info}/WHEEL +0 -0
  39. {dls_dodal-1.48.0.dist-info → dls_dodal-1.50.0.dist-info}/entry_points.txt +0 -0
  40. {dls_dodal-1.48.0.dist-info → dls_dodal-1.50.0.dist-info}/licenses/LICENSE +0 -0
  41. {dls_dodal-1.48.0.dist-info → dls_dodal-1.50.0.dist-info}/top_level.txt +0 -0
@@ -6,6 +6,7 @@ from ophyd_async.core import (
6
6
  AsyncStatus,
7
7
  LazyMock,
8
8
  SignalR,
9
+ SignalRW,
9
10
  StandardReadable,
10
11
  derived_signal_r,
11
12
  soft_signal_rw,
@@ -36,7 +37,21 @@ def _get_correct_zoom_string(zoom: str) -> str:
36
37
  return zoom
37
38
 
38
39
 
39
- class ZoomController(StandardReadable, Movable[str]):
40
+ class BaseZoomController(StandardReadable, Movable[str]):
41
+ level: SignalRW[str]
42
+ percentage: SignalRW[float]
43
+
44
+
45
+ class NullZoomController(BaseZoomController):
46
+ def __init__(self):
47
+ self.level = soft_signal_rw(str, "1.0x")
48
+ self.percentage = soft_signal_rw(float, 100)
49
+
50
+ def set(self, value):
51
+ raise Exception("Attempting to set zoom level of a null zoom controller")
52
+
53
+
54
+ class ZoomController(BaseZoomController):
40
55
  """
41
56
  Device to control the zoom level. This should be set like
42
57
  o = OAV(name="oav")
@@ -62,12 +77,24 @@ class OAV(StandardReadable):
62
77
  beam_centre_i: SignalR[int]
63
78
  beam_centre_j: SignalR[int]
64
79
 
65
- def __init__(self, prefix: str, config: OAVConfigBase, name: str = ""):
80
+ def __init__(
81
+ self,
82
+ prefix: str,
83
+ config: OAVConfigBase,
84
+ name: str = "",
85
+ zoom_controller: BaseZoomController | None = None,
86
+ ):
66
87
  self.oav_config = config
67
88
  self._prefix = prefix
68
89
  self._name = name
69
90
  _bl_prefix = prefix.split("-")[0]
70
- self.zoom_controller = ZoomController(f"{_bl_prefix}-EA-OAV-01:FZOOM:", name)
91
+
92
+ if not zoom_controller:
93
+ self.zoom_controller = ZoomController(
94
+ f"{_bl_prefix}-EA-OAV-01:FZOOM:", name
95
+ )
96
+ else:
97
+ self.zoom_controller = zoom_controller
71
98
 
72
99
  self.cam = Cam(f"{prefix}CAM:", name=name)
73
100
  with self.add_children_as_readables():
@@ -121,8 +148,14 @@ class OAVBeamCentreFile(OAV):
121
148
  centre values are stored.
122
149
  """
123
150
 
124
- def __init__(self, prefix: str, config: OAVConfigBeamCentre, name: str = ""):
125
- super().__init__(prefix, config, name)
151
+ def __init__(
152
+ self,
153
+ prefix: str,
154
+ config: OAVConfigBeamCentre,
155
+ name: str = "",
156
+ zoom_controller: BaseZoomController | None = None,
157
+ ):
158
+ super().__init__(prefix, config, name, zoom_controller)
126
159
 
127
160
  with self.add_children_as_readables():
128
161
  self.beam_centre_i = derived_signal_r(
@@ -152,7 +185,12 @@ class OAVBeamCentrePV(OAV):
152
185
  """OAV device that reads its beam centre values from PVs."""
153
186
 
154
187
  def __init__(
155
- self, prefix: str, config: OAVConfig, name: str = "", overlay_channel: int = 1
188
+ self,
189
+ prefix: str,
190
+ config: OAVConfig,
191
+ name: str = "",
192
+ zoom_controller: BaseZoomController | None = None,
193
+ overlay_channel: int = 1,
156
194
  ):
157
195
  with self.add_children_as_readables():
158
196
  self.beam_centre_i = epics_signal_r(
@@ -161,4 +199,4 @@ class OAVBeamCentrePV(OAV):
161
199
  self.beam_centre_j = epics_signal_r(
162
200
  int, prefix + f"OVER:{overlay_channel}:CenterY"
163
201
  )
164
- super().__init__(prefix, config, name)
202
+ super().__init__(prefix, config, name, zoom_controller)
dodal/plans/bimorph.py ADDED
@@ -0,0 +1,333 @@
1
+ from collections.abc import Generator
2
+ from dataclasses import dataclass
3
+ from enum import Enum
4
+ from typing import Any
5
+
6
+ import bluesky.plan_stubs as bps
7
+ import bluesky.preprocessors as bpp
8
+ from bluesky.protocols import Preparable, Readable
9
+ from bluesky.utils import MsgGenerator
10
+ from numpy import linspace
11
+ from ophyd_async.core import TriggerInfo
12
+
13
+ from dodal.devices.bimorph_mirror import BimorphMirror
14
+ from dodal.devices.slits import Slits
15
+
16
+
17
+ class SlitDimension(str, Enum):
18
+ """Enum representing the dimensions of a 2d slit
19
+
20
+ Used to describe which dimension the pencil beam scan should move across.
21
+ The other dimension will be held constant.
22
+
23
+ Attributes:
24
+ X: Represents X dimension
25
+ Y: Represents Y dimension
26
+ """
27
+
28
+ X = "X"
29
+ Y = "Y"
30
+
31
+
32
+ def move_slits(slits: Slits, dimension: SlitDimension, gap: float, center: float):
33
+ """Moves ones dimension of Slits object to given position.
34
+
35
+ Args:
36
+ slits: Slits to move
37
+ dimension: SlitDimension (X or Y)
38
+ gap: float size of gap
39
+ center: float position of center
40
+ """
41
+ if dimension == SlitDimension.X:
42
+ yield from bps.mv(slits.x_gap, gap)
43
+ yield from bps.mv(slits.x_centre, center)
44
+ else:
45
+ yield from bps.mv(slits.y_gap, gap)
46
+ yield from bps.mv(slits.y_centre, center)
47
+
48
+
49
+ def check_valid_bimorph_state(
50
+ voltage_list: list[float], abs_range: float, abs_diff: float
51
+ ) -> bool:
52
+ """Checks that a set of bimorph voltages is valid.
53
+ Args:
54
+ voltage_list: float amount each actuator will be increased by per scan
55
+ abs_range: float absolute value of maximum possible voltage of each actuator
56
+ abs_diff: float absolute maximum difference between two consecutive actuators
57
+
58
+ Returns:
59
+ Bool representing state validity
60
+ """
61
+ for voltage in voltage_list:
62
+ if abs(voltage) > abs_range:
63
+ return False
64
+
65
+ for i in range(len(voltage_list) - 1):
66
+ if abs(voltage_list[i] - voltage_list[i - 1]) > abs_diff:
67
+ return False
68
+
69
+ return True
70
+
71
+
72
+ def validate_bimorph_plan(
73
+ initial_voltage_list: list[float],
74
+ voltage_increment: float,
75
+ abs_range: float,
76
+ abs_diff: float,
77
+ ) -> bool:
78
+ """Checks that every position the bimorph will move through will not error.
79
+
80
+ Args:
81
+ initial_voltage_list: float list starting position
82
+ voltage_increment: float amount each actuator will be increased by per scan
83
+ abs_range: float absolute value of maximum possible voltage of each actuator
84
+ abs_diff: float absolute maximum difference between two consecutive actuators
85
+
86
+ Raises:
87
+ Exception if the plan will lead to an error state"""
88
+ voltage_list = initial_voltage_list.copy()
89
+
90
+ if not check_valid_bimorph_state(voltage_list, abs_range, abs_diff):
91
+ raise ValueError(f"Bimorph plan reaches invalid state at: {voltage_list}")
92
+
93
+ for i in range(len(initial_voltage_list)):
94
+ voltage_list[i] += voltage_increment
95
+
96
+ if not check_valid_bimorph_state(voltage_list, abs_range, abs_diff):
97
+ raise ValueError(f"Bimorph plan reaches invalid state at: {voltage_list}")
98
+
99
+ return True
100
+
101
+
102
+ @dataclass
103
+ class BimorphState:
104
+ """Data class containing positions of BimorphMirror and Slits"""
105
+
106
+ voltages: list[float]
107
+ x_gap: float
108
+ y_gap: float
109
+ x_center: float
110
+ y_center: float
111
+
112
+
113
+ def capture_bimorph_state(mirror: BimorphMirror, slits: Slits):
114
+ """Plan stub that captures current position of BimorphMirror and Slits.
115
+
116
+ Args:
117
+ mirror: BimorphMirror to read from
118
+ slits: Slits to read from
119
+
120
+ Returns:
121
+ A BimorphState containing BimorphMirror and Slits positions"""
122
+ original_voltage_list = []
123
+
124
+ for channel in mirror.channels.values():
125
+ position = yield from bps.rd(channel.output_voltage)
126
+ original_voltage_list.append(position)
127
+
128
+ original_x_gap = yield from bps.rd(slits.x_gap)
129
+ original_y_gap = yield from bps.rd(slits.y_gap)
130
+ original_x_center = yield from bps.rd(slits.x_centre)
131
+ original_y_center = yield from bps.rd(slits.y_centre)
132
+ return BimorphState(
133
+ original_voltage_list,
134
+ original_x_gap,
135
+ original_y_gap,
136
+ original_x_center,
137
+ original_y_center,
138
+ )
139
+
140
+
141
+ def restore_bimorph_state(mirror: BimorphMirror, slits: Slits, state: BimorphState):
142
+ """Moves BimorphMirror and Slits to state given in BirmophState.
143
+
144
+ Args:
145
+ mirror: BimorphMirror to move
146
+ slits: Slits to move
147
+ state: BimorphState to move to.
148
+ """
149
+ yield from move_slits(slits, SlitDimension.X, state.x_gap, state.x_center)
150
+ yield from move_slits(slits, SlitDimension.Y, state.y_gap, state.y_center)
151
+
152
+ yield from bps.mv(mirror, state.voltages) # type: ignore
153
+
154
+
155
+ def bimorph_position_generator(
156
+ initial_voltage_list: list[float], voltage_increment: float
157
+ ) -> Generator[list[float], None, None]:
158
+ """Generator that produces bimorph positions, starting with the initial_voltage_list.
159
+
160
+ Args:
161
+ initial_voltage_list: list starting position for bimorph
162
+ voltage_increment: float amount to increase each actuator by in turn
163
+
164
+ Yields:
165
+ List bimorph positions, starting with initial_voltage_list
166
+ """
167
+ voltage_list = initial_voltage_list.copy()
168
+
169
+ for i in range(-1, len(initial_voltage_list)):
170
+ yield [
171
+ voltage + voltage_increment if i >= j else voltage
172
+ for (j, voltage) in enumerate(voltage_list)
173
+ ]
174
+
175
+
176
+ def bimorph_optimisation(
177
+ detectors: list[Readable],
178
+ mirror: BimorphMirror,
179
+ slits: Slits,
180
+ voltage_increment: float,
181
+ active_dimension: SlitDimension,
182
+ active_slit_center_start: float,
183
+ active_slit_center_end: float,
184
+ active_slit_size: float,
185
+ inactive_slit_center: float,
186
+ inactive_slit_size: float,
187
+ number_of_slit_positions: int,
188
+ bimorph_settle_time: float,
189
+ slit_settle_time: float,
190
+ initial_voltage_list: list | None = None,
191
+ metadata: dict[str, Any] | None = None,
192
+ ) -> MsgGenerator:
193
+ """Plan for performing bimorph mirror optimisation.
194
+
195
+ Bluesky plan that performs a series of pencil beam scans across one axis of a
196
+ bimorph mirror, of using a 2-dimensional slit.
197
+
198
+ Args:
199
+ detectors: list[Readable] detectors
200
+ bimorph: BimorphMirror to move
201
+ slit: Slits
202
+ voltage_increment: float voltage increment applied to each bimorph electrode
203
+ active_dimension: SlitDimension that slit will move in (X or Y)
204
+ active_slit_center_start: float start position of center of slit in active dimension
205
+ active_slit_center_end: float final position of center of slit in active dimension
206
+ active_slit_size: float size of slit in active dimension
207
+ inactive_slit_center: float center of slit in inactive dimension
208
+ inactive_slit_size: float size of slit in inactive dimension
209
+ number_of_slit_positions: int number of slit positions per pencil beam scan
210
+ bimorph_settle_time: float time in seconds to wait after bimorph move
211
+ slit_settle_time: float time in seconds to wait after slit move
212
+ initial_voltage_list: optional list[float] starting voltages for bimorph (defaults to current voltages)
213
+ metadata: optional dict[str, Any] metadata to add to start document
214
+ """
215
+
216
+ _metadata = {
217
+ "plan_args": {
218
+ "detectors": {det.name for det in detectors},
219
+ "mirror": mirror.name,
220
+ "slits": slits.name,
221
+ "voltage_increment": voltage_increment,
222
+ "active_dimension": active_dimension,
223
+ "active_slit_center_start": active_slit_center_start,
224
+ "active_slit_center_end": active_slit_center_end,
225
+ "active_slit_size": active_slit_size,
226
+ "inactive_slit_center": inactive_slit_center,
227
+ "inactive_slit_size": inactive_slit_size,
228
+ "number_of_slit_positions": number_of_slit_positions,
229
+ "bimorph_settle_time": bimorph_settle_time,
230
+ "slit_settle_time": slit_settle_time,
231
+ "initial_voltage_list": initial_voltage_list,
232
+ },
233
+ "plan_name": "bimorph_optimisation",
234
+ "shape": [len(mirror.channels), number_of_slit_positions],
235
+ **(metadata or {}),
236
+ }
237
+
238
+ state = yield from capture_bimorph_state(mirror, slits)
239
+
240
+ # If a starting set of voltages is not provided, default to current:
241
+ initial_voltage_list = initial_voltage_list or state.voltages
242
+
243
+ bimorph_positions = bimorph_position_generator(
244
+ initial_voltage_list, voltage_increment
245
+ )
246
+
247
+ validate_bimorph_plan(initial_voltage_list, voltage_increment, 1000, 500)
248
+
249
+ inactive_dimension = (
250
+ SlitDimension.Y if active_dimension == SlitDimension.X else SlitDimension.X
251
+ )
252
+
253
+ @bpp.run_decorator(md=_metadata)
254
+ @bpp.stage_decorator((*detectors, mirror, slits))
255
+ def outer_scan():
256
+ """Outer plan stub, which moves mirror and calls inner_scan."""
257
+ for detector in detectors:
258
+ if isinstance(detector, Preparable):
259
+ yield from bps.prepare(detector, TriggerInfo(), wait=True)
260
+
261
+ stream_name = "0"
262
+ yield from bps.declare_stream(*detectors, mirror, slits, name=stream_name)
263
+
264
+ # Move slits into starting position:
265
+ yield from move_slits(
266
+ slits, active_dimension, active_slit_size, active_slit_center_start
267
+ )
268
+ yield from move_slits(
269
+ slits, inactive_dimension, inactive_slit_size, inactive_slit_center
270
+ )
271
+ yield from bps.sleep(slit_settle_time)
272
+
273
+ for bimorph_position in bimorph_positions:
274
+ yield from bps.mv(
275
+ mirror, # type: ignore
276
+ bimorph_position, # type: ignore
277
+ )
278
+ yield from bps.sleep(bimorph_settle_time)
279
+
280
+ yield from bps.declare_stream(*detectors, mirror, slits, name=stream_name)
281
+
282
+ yield from inner_scan(
283
+ detectors,
284
+ mirror,
285
+ slits,
286
+ active_dimension,
287
+ active_slit_center_start,
288
+ active_slit_center_end,
289
+ active_slit_size,
290
+ number_of_slit_positions,
291
+ slit_settle_time,
292
+ stream_name,
293
+ )
294
+
295
+ stream_name = str(int(stream_name) + 1)
296
+
297
+ yield from outer_scan()
298
+
299
+ yield from restore_bimorph_state(mirror, slits, state)
300
+
301
+
302
+ def inner_scan(
303
+ detectors: list[Readable],
304
+ mirror: BimorphMirror,
305
+ slits: Slits,
306
+ active_dimension: SlitDimension,
307
+ active_slit_center_start: float,
308
+ active_slit_center_end: float,
309
+ active_slit_size: float,
310
+ number_of_slit_positions: int,
311
+ slit_settle_time: float,
312
+ stream_name: str,
313
+ ):
314
+ """Inner plan stub, which moves Slits and performs a read.
315
+
316
+ Args:
317
+ mirror: BimorphMirror to move
318
+ slit: Slits
319
+ oav: oav on-axis viewer
320
+ active_dimension: SlitDimension that slit will move in (X or Y)
321
+ active_slit_center_start: float start position of center of slit in active dimension
322
+ active_slit_center_end: float final position of center of slit in active dimension
323
+ active_slit_size: float size of slit in active dimension
324
+ number_of_slit_positions: int number of slit positions per pencil beam scan
325
+ slit_settle_time: float time in seconds to wait after slit move
326
+ stream_name: str name to pass to trigger_and_read
327
+ """
328
+ for value in linspace(
329
+ active_slit_center_start, active_slit_center_end, number_of_slit_positions
330
+ ):
331
+ yield from move_slits(slits, active_dimension, active_slit_size, value)
332
+ yield from bps.sleep(slit_settle_time)
333
+ yield from bps.trigger_and_read([*detectors, mirror, slits], name=stream_name)