ophyd-async 0.2.0__py3-none-any.whl → 0.3a2__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 (62) hide show
  1. ophyd_async/__init__.py +1 -4
  2. ophyd_async/_version.py +2 -2
  3. ophyd_async/core/__init__.py +16 -10
  4. ophyd_async/core/_providers.py +38 -5
  5. ophyd_async/core/async_status.py +3 -3
  6. ophyd_async/core/detector.py +211 -62
  7. ophyd_async/core/device.py +45 -38
  8. ophyd_async/core/device_save_loader.py +96 -23
  9. ophyd_async/core/flyer.py +30 -244
  10. ophyd_async/core/signal.py +47 -21
  11. ophyd_async/core/signal_backend.py +7 -4
  12. ophyd_async/core/sim_signal_backend.py +30 -18
  13. ophyd_async/core/standard_readable.py +4 -2
  14. ophyd_async/core/utils.py +93 -30
  15. ophyd_async/epics/_backend/_aioca.py +30 -36
  16. ophyd_async/epics/_backend/_p4p.py +75 -41
  17. ophyd_async/epics/_backend/common.py +25 -0
  18. ophyd_async/epics/areadetector/__init__.py +4 -0
  19. ophyd_async/epics/areadetector/aravis.py +69 -0
  20. ophyd_async/epics/areadetector/controllers/ad_sim_controller.py +1 -1
  21. ophyd_async/epics/areadetector/controllers/aravis_controller.py +73 -0
  22. ophyd_async/epics/areadetector/controllers/pilatus_controller.py +37 -25
  23. ophyd_async/epics/areadetector/drivers/aravis_driver.py +154 -0
  24. ophyd_async/epics/areadetector/drivers/pilatus_driver.py +4 -4
  25. ophyd_async/epics/areadetector/pilatus.py +50 -0
  26. ophyd_async/epics/areadetector/writers/_hdffile.py +21 -7
  27. ophyd_async/epics/areadetector/writers/hdf_writer.py +26 -15
  28. ophyd_async/epics/demo/__init__.py +33 -3
  29. ophyd_async/epics/motion/motor.py +20 -14
  30. ophyd_async/epics/pvi/__init__.py +3 -0
  31. ophyd_async/epics/pvi/pvi.py +318 -0
  32. ophyd_async/epics/signal/__init__.py +0 -2
  33. ophyd_async/epics/signal/signal.py +26 -9
  34. ophyd_async/panda/__init__.py +19 -5
  35. ophyd_async/panda/_common_blocks.py +49 -0
  36. ophyd_async/panda/_hdf_panda.py +48 -0
  37. ophyd_async/panda/_panda_controller.py +37 -0
  38. ophyd_async/panda/_trigger.py +39 -0
  39. ophyd_async/panda/_utils.py +15 -0
  40. ophyd_async/panda/writers/__init__.py +3 -0
  41. ophyd_async/panda/writers/_hdf_writer.py +220 -0
  42. ophyd_async/panda/writers/_panda_hdf_file.py +58 -0
  43. ophyd_async/planstubs/__init__.py +5 -0
  44. ophyd_async/planstubs/prepare_trigger_and_dets.py +57 -0
  45. ophyd_async/protocols.py +73 -0
  46. ophyd_async/sim/__init__.py +11 -0
  47. ophyd_async/sim/demo/__init__.py +3 -0
  48. ophyd_async/sim/demo/sim_motor.py +116 -0
  49. ophyd_async/sim/pattern_generator.py +318 -0
  50. ophyd_async/sim/sim_pattern_detector_control.py +55 -0
  51. ophyd_async/sim/sim_pattern_detector_writer.py +34 -0
  52. ophyd_async/sim/sim_pattern_generator.py +37 -0
  53. {ophyd_async-0.2.0.dist-info → ophyd_async-0.3a2.dist-info}/METADATA +20 -76
  54. ophyd_async-0.3a2.dist-info/RECORD +76 -0
  55. {ophyd_async-0.2.0.dist-info → ophyd_async-0.3a2.dist-info}/WHEEL +1 -1
  56. ophyd_async/epics/signal/pvi_get.py +0 -22
  57. ophyd_async/panda/panda.py +0 -294
  58. ophyd_async-0.2.0.dist-info/RECORD +0 -53
  59. /ophyd_async/panda/{table.py → _table.py} +0 -0
  60. {ophyd_async-0.2.0.dist-info → ophyd_async-0.3a2.dist-info}/LICENSE +0 -0
  61. {ophyd_async-0.2.0.dist-info → ophyd_async-0.3a2.dist-info}/entry_points.txt +0 -0
  62. {ophyd_async-0.2.0.dist-info → ophyd_async-0.3a2.dist-info}/top_level.txt +0 -0
@@ -1,22 +0,0 @@
1
- from typing import Dict, TypedDict
2
-
3
- from p4p.client.asyncio import Context
4
-
5
-
6
- class PVIEntry(TypedDict, total=False):
7
- d: str
8
- r: str
9
- rw: str
10
- w: str
11
- x: str
12
-
13
-
14
- async def pvi_get(pv: str, ctxt: Context, timeout: float = 5.0) -> Dict[str, PVIEntry]:
15
- pv_info = ctxt.get(pv, timeout=timeout).get("pvi").todict()
16
-
17
- result = {}
18
-
19
- for attr_name, attr_info in pv_info.items():
20
- result[attr_name] = PVIEntry(**attr_info) # type: ignore
21
-
22
- return result
@@ -1,294 +0,0 @@
1
- from __future__ import annotations
2
-
3
- import atexit
4
- import re
5
- from typing import (
6
- Callable,
7
- Dict,
8
- FrozenSet,
9
- Optional,
10
- Tuple,
11
- Type,
12
- TypedDict,
13
- cast,
14
- get_args,
15
- get_origin,
16
- get_type_hints,
17
- )
18
-
19
- from p4p.client.thread import Context
20
-
21
- from ophyd_async.core import (
22
- Device,
23
- DeviceVector,
24
- Signal,
25
- SignalBackend,
26
- SignalR,
27
- SignalRW,
28
- SignalX,
29
- SimSignalBackend,
30
- )
31
- from ophyd_async.epics.signal import (
32
- epics_signal_r,
33
- epics_signal_rw,
34
- epics_signal_w,
35
- epics_signal_x,
36
- pvi_get,
37
- )
38
- from ophyd_async.panda.table import SeqTable
39
-
40
-
41
- class PulseBlock(Device):
42
- delay: SignalRW[float]
43
- width: SignalRW[float]
44
-
45
-
46
- class SeqBlock(Device):
47
- table: SignalRW[SeqTable]
48
- active: SignalRW[bool]
49
-
50
-
51
- class PcapBlock(Device):
52
- active: SignalR[bool]
53
-
54
-
55
- class PVIEntry(TypedDict, total=False):
56
- d: str
57
- r: str
58
- rw: str
59
- w: str
60
- x: str
61
-
62
-
63
- def _block_name_number(block_name: str) -> Tuple[str, Optional[int]]:
64
- """Maps a panda block name to a block and number.
65
-
66
- There are exceptions to this rule; some blocks like pcap do not contain numbers.
67
- Other blocks may contain numbers and letters, but no numbers at the end.
68
-
69
- Such block names will only return the block name, and not a number.
70
-
71
- If this function returns both a block name and number, it should be instantiated
72
- into a device vector."""
73
- m = re.match("^([0-9a-z_-]*)([0-9]+)$", block_name)
74
- if m is not None:
75
- name, num = m.groups()
76
- return name, int(num or 1) # just to pass type checks.
77
-
78
- return block_name, None
79
-
80
-
81
- def _remove_inconsistent_blocks(pvi_info: Dict[str, PVIEntry]) -> None:
82
- """Remove blocks from pvi information.
83
-
84
- This is needed because some pandas have 'pcap' and 'pcap1' blocks, which are
85
- inconsistent with the assumption that pandas should only have a 'pcap' block,
86
- for example.
87
-
88
- """
89
- pvi_keys = set(pvi_info.keys())
90
- for k in pvi_keys:
91
- kn = re.sub(r"\d*$", "", k)
92
- if kn and k != kn and kn in pvi_keys:
93
- del pvi_info[k]
94
-
95
-
96
- async def pvi(pv: str, ctxt: Context, timeout: float = 5.0) -> Dict[str, PVIEntry]:
97
- result = await pvi_get(pv, ctxt, timeout=timeout)
98
- _remove_inconsistent_blocks(result)
99
- return result
100
-
101
-
102
- class PandA(Device):
103
- _ctxt: Optional[Context] = None
104
-
105
- pulse: DeviceVector[PulseBlock]
106
- seq: DeviceVector[SeqBlock]
107
- pcap: PcapBlock
108
-
109
- def __init__(self, pv: str) -> None:
110
- self._init_prefix = pv
111
- self.pvi_mapping: Dict[FrozenSet[str], Callable[..., Signal]] = {
112
- frozenset({"r", "w"}): lambda dtype, rpv, wpv: epics_signal_rw(
113
- dtype, rpv, wpv
114
- ),
115
- frozenset({"rw"}): lambda dtype, rpv, wpv: epics_signal_rw(dtype, rpv, wpv),
116
- frozenset({"r"}): lambda dtype, rpv, wpv: epics_signal_r(dtype, rpv),
117
- frozenset({"w"}): lambda dtype, rpv, wpv: epics_signal_w(dtype, wpv),
118
- frozenset({"x"}): lambda dtype, rpv, wpv: epics_signal_x(wpv),
119
- }
120
-
121
- @property
122
- def ctxt(self) -> Context:
123
- if PandA._ctxt is None:
124
- PandA._ctxt = Context("pva", nt=False)
125
-
126
- @atexit.register
127
- def _del_ctxt():
128
- # If we don't do this we get messages like this on close:
129
- # Error in sys.excepthook:
130
- # Original exception was:
131
- PandA._ctxt = None
132
-
133
- return PandA._ctxt
134
-
135
- def verify_block(self, name: str, num: Optional[int]):
136
- """Given a block name and number, return information about a block."""
137
- anno = get_type_hints(self, globalns=globals()).get(name)
138
-
139
- block: Device = Device()
140
-
141
- if anno:
142
- type_args = get_args(anno)
143
- block = type_args[0]() if type_args else anno()
144
-
145
- if not type_args:
146
- assert num is None, f"Only expected one {name} block, got {num}"
147
-
148
- return block
149
-
150
- async def _make_block(
151
- self, name: str, num: Optional[int], block_pv: str, sim: bool = False
152
- ):
153
- """Makes a block given a block name containing relevant signals.
154
-
155
- Loops through the signals in the block (found using type hints), if not in
156
- sim mode then does a pvi call, and identifies this signal from the pvi call.
157
- """
158
- block = self.verify_block(name, num)
159
-
160
- field_annos = get_type_hints(block, globalns=globals())
161
- block_pvi = await pvi(block_pv, self.ctxt) if not sim else None
162
-
163
- # finds which fields this class actually has, e.g. delay, width...
164
- for sig_name, sig_type in field_annos.items():
165
- origin = get_origin(sig_type)
166
- args = get_args(sig_type)
167
-
168
- # if not in sim mode,
169
- if block_pvi:
170
- block_pvi = cast(Dict[str, PVIEntry], block_pvi)
171
- # try to get this block in the pvi.
172
- entry: Optional[PVIEntry] = block_pvi.get(sig_name)
173
- if entry is None:
174
- raise Exception(
175
- f"{self.__class__.__name__} has a {name} block containing a/"
176
- + f"an {sig_name} signal which has not been retrieved by PVI."
177
- )
178
-
179
- signal = self._make_signal(entry, args[0] if len(args) > 0 else None)
180
-
181
- else:
182
- backend: SignalBackend = SimSignalBackend(
183
- args[0] if len(args) > 0 else None, block_pv
184
- )
185
- signal = SignalX(backend) if not origin else origin(backend)
186
-
187
- setattr(block, sig_name, signal)
188
-
189
- # checks for any extra pvi information not contained in this class
190
- if block_pvi:
191
- for attr, attr_pvi in block_pvi.items():
192
- if not hasattr(block, attr):
193
- # makes any extra signals
194
- signal = self._make_signal(attr_pvi)
195
- setattr(block, attr, signal)
196
-
197
- return block
198
-
199
- async def _make_untyped_block(self, block_pv: str):
200
- """Populates a block using PVI information.
201
-
202
- This block is not typed as part of the PandA interface but needs to be
203
- included dynamically anyway.
204
- """
205
- block = Device()
206
- block_pvi: Dict[str, PVIEntry] = await pvi(block_pv, self.ctxt)
207
-
208
- for signal_name, signal_pvi in block_pvi.items():
209
- signal = self._make_signal(signal_pvi)
210
- setattr(block, signal_name, signal)
211
-
212
- return block
213
-
214
- def _make_signal(self, signal_pvi: PVIEntry, dtype: Optional[Type] = None):
215
- """Make a signal.
216
-
217
- This assumes datatype is None so it can be used to create dynamic signals.
218
- """
219
- operations = frozenset(signal_pvi.keys())
220
- pvs = [signal_pvi[i] for i in operations] # type: ignore
221
- signal_factory = self.pvi_mapping[operations]
222
-
223
- write_pv = pvs[0]
224
- read_pv = write_pv if len(pvs) == 1 else pvs[1]
225
-
226
- return signal_factory(dtype, "pva://" + read_pv, "pva://" + write_pv)
227
-
228
- # TODO redo to set_panda_block? confusing name
229
- def set_attribute(self, name: str, num: Optional[int], block: Device):
230
- """Set a block on the panda.
231
-
232
- Need to be able to set device vectors on the panda as well, e.g. if num is not
233
- None, need to be able to make a new device vector and start populating it...
234
- """
235
- anno = get_type_hints(self, globalns=globals()).get(name)
236
-
237
- # if it's an annotated device vector, or it isn't but we've got a number then
238
- # make a DeviceVector on the class
239
- if get_origin(anno) == DeviceVector or (not anno and num is not None):
240
- self.__dict__.setdefault(name, DeviceVector())[num] = block
241
- else:
242
- setattr(self, name, block)
243
-
244
- async def connect(self, sim=False) -> None:
245
- """Initialises all blocks and connects them.
246
-
247
- First, checks for pvi information. If it exists, make all blocks from this.
248
- Then, checks that all required blocks in the PandA have been made.
249
-
250
- If there's no pvi information, that's because we're in sim mode. In that case,
251
- makes all required blocks.
252
- """
253
- pvi_info = await pvi(self._init_prefix + ":PVI", self.ctxt) if not sim else None
254
- hints = {
255
- attr_name: attr_type
256
- for attr_name, attr_type in get_type_hints(self, globalns=globals()).items()
257
- if not attr_name.startswith("_")
258
- }
259
-
260
- # create all the blocks pvi says it should have,
261
- if pvi_info:
262
- pvi_info = cast(Dict[str, PVIEntry], pvi_info)
263
- for block_name, block_pvi in pvi_info.items():
264
- name, num = _block_name_number(block_name)
265
-
266
- if name in hints:
267
- block = await self._make_block(name, num, block_pvi["d"])
268
- else:
269
- block = await self._make_untyped_block(block_pvi["d"])
270
-
271
- self.set_attribute(name, num, block)
272
-
273
- # then check if the ones defined in this class are in the pvi info
274
- # make them if there is no pvi info, i.e. sim mode.
275
- for block_name in hints.keys():
276
- if pvi_info is not None:
277
- pvi_name = block_name
278
-
279
- if get_origin(hints[block_name]) == DeviceVector:
280
- pvi_name += "1"
281
-
282
- entry: Optional[PVIEntry] = pvi_info.get(pvi_name)
283
-
284
- assert entry, f"Expected PandA to contain {block_name} block."
285
- assert list(entry) == [
286
- "d"
287
- ], f"Expected PandA to only contain blocks, got {entry}"
288
- else:
289
- num = 1 if get_origin(hints[block_name]) == DeviceVector else None
290
- block = await self._make_block(block_name, num, "sim://", sim=sim)
291
- self.set_attribute(block_name, num, block)
292
-
293
- self.set_name(self.name)
294
- await super().connect(sim)
@@ -1,53 +0,0 @@
1
- ophyd_async/__init__.py,sha256=WJoRU7gO-hRzyf7a-C952zF6-7zwPjP7qZ1Qu5GTUC8,124
2
- ophyd_async/__main__.py,sha256=G-Zcv_G9zK7Nhx6o5L5w-wyhMxdl_WgyMELu8IMFqAE,328
3
- ophyd_async/_version.py,sha256=H-qsvrxCpdhaQzyddR-yajEqI71hPxLa4KxzpP3uS1g,411
4
- ophyd_async/core/__init__.py,sha256=Us5DMr4ys-P4_zKWs6rN-J8QipWpOjAsmF5Cu0BY9k8,2188
5
- ophyd_async/core/_providers.py,sha256=sgEv6MabSKlrFwB8TEoTU7qKiQ6D1YsO33q2E3oKK_M,940
6
- ophyd_async/core/async_status.py,sha256=j3sE2zCvfg0GZWVY1NCEhdkdrqGhwtoDqwX_SVnFhbA,2765
7
- ophyd_async/core/detector.py,sha256=seeokcIRBj-maUuODUOKxesXRJjZLuEhOh6BzDmdIQI,6375
8
- ophyd_async/core/device.py,sha256=4PCvf7MU91CiWvJf0ERWEWE5wu7XPNrUND1ScxzMOXY,5958
9
- ophyd_async/core/device_save_loader.py,sha256=oAxWleSSOUD5TrnZWjOOmsVEfk2dE0cOvdwLTkEifvQ,6935
10
- ophyd_async/core/flyer.py,sha256=n3lqcvfXJiZooXXJocvaN6Wzi3wsqIPKzG4jD5veO7o,9970
11
- ophyd_async/core/signal.py,sha256=2szVlaxdZwc5z8gtnH8Yaol1jQIMrNMwhNWlvdZtwwI,11589
12
- ophyd_async/core/signal_backend.py,sha256=brD1MTug_AkNj3hDkhz6CxG5ygE1LF4AS9YVWCMFI0M,1312
13
- ophyd_async/core/sim_signal_backend.py,sha256=XZ2NCgoLnVPJyqwhRRhuUjY4TRXrVl6zH-5UAVLJGvY,5539
14
- ophyd_async/core/standard_readable.py,sha256=9cBetIYttAho-7wOB3T1YgSJy0iWEFTAdn9jZkrGvLA,2560
15
- ophyd_async/core/utils.py,sha256=uxmKSbLsRizna482Zsl2q14H_jiM3JksqHCFkp_QYIM,3193
16
- ophyd_async/epics/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
17
- ophyd_async/epics/_backend/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
18
- ophyd_async/epics/_backend/_aioca.py,sha256=izEUbxHKRLalsbcfybmMKUHw7NvDOvST4lqzIRxlOwg,9033
19
- ophyd_async/epics/_backend/_p4p.py,sha256=vrCjX2W6mkghZfyCFI_i5VLhG3380nyW1-HiupzRjKQ,11001
20
- ophyd_async/epics/areadetector/__init__.py,sha256=oEOzL7gEVVSULeT2WkFrva8lUKeX0zmQD0tmwFz6L24,325
21
- ophyd_async/epics/areadetector/single_trigger_det.py,sha256=q5mG-OUVagIjvXLb28lsrGj4eUSoH2pNW2rT4rQR8fA,1206
22
- ophyd_async/epics/areadetector/utils.py,sha256=dez54oElIkGMnhSM9qghToUB1opSqjdWTV2vhHCgRMA,3133
23
- ophyd_async/epics/areadetector/controllers/__init__.py,sha256=UG2-M5d2ykp2T8isQJCbAsGZF1aH0BtC_OPlzzPTjnA,149
24
- ophyd_async/epics/areadetector/controllers/ad_sim_controller.py,sha256=Q5GT7nzXWL3MxJ25b6rab_qft-XQtwGEfrGG9dNKJcA,1591
25
- ophyd_async/epics/areadetector/controllers/pilatus_controller.py,sha256=n9Gkq2RtPnTzdoqpKArvod92Nu1Jo7XHJBGUkZr-N2s,1556
26
- ophyd_async/epics/areadetector/drivers/__init__.py,sha256=AOpIEYfoBhG9Nc4-SId99v4PpyEh4_RBXfNaqiXlwUI,315
27
- ophyd_async/epics/areadetector/drivers/ad_base.py,sha256=ikfyNcZwJa5ah52DckjrBzkMMT_eDY1smM4XWfb6A6E,3689
28
- ophyd_async/epics/areadetector/drivers/pilatus_driver.py,sha256=cn5WNz913UOnOttw2bssjV2Bo3p9SuJma3ckRyCdvw8,442
29
- ophyd_async/epics/areadetector/writers/__init__.py,sha256=tpPcrYd1hs8WS7C0gmCnR2EBwjE5RzCljI7WwZ2V_LM,191
30
- ophyd_async/epics/areadetector/writers/_hdfdataset.py,sha256=E0C9VgsPyY35h7k0mvcIhjsIVNavApLxizqNWlM388w,167
31
- ophyd_async/epics/areadetector/writers/_hdffile.py,sha256=bFVDlVIMXANtE146DCm4uVbnYaibQM8Qzoa32XcwMJg,1373
32
- ophyd_async/epics/areadetector/writers/hdf_writer.py,sha256=_JAU6CSOigxi7SW6Wbu1jX7__8U-03HkjaPLF5VfHfQ,4666
33
- ophyd_async/epics/areadetector/writers/nd_file_hdf.py,sha256=rutCstILCGGwhP5pH_2lWM2QUcZ88-uxx5dTZIJUMWQ,1562
34
- ophyd_async/epics/areadetector/writers/nd_plugin.py,sha256=l0yBBEazviyFsWJv_4_sfGn_YM_Iyd0_SlMdAmUlXDU,871
35
- ophyd_async/epics/demo/__init__.py,sha256=DbVO4ufJWjQnZteilW8SBs5A8DN-Xajn0YibM0q8UkE,5500
36
- ophyd_async/epics/demo/demo_ad_sim_detector.py,sha256=06y65yvaqXvL2rDocjYyLz9kTVzuwV-LeuPhEfExdOA,944
37
- ophyd_async/epics/demo/mover.db,sha256=RFz0rxZue689Wh1sWTZwWeFMUrH04ttPq2u5xJH_Fp4,998
38
- ophyd_async/epics/demo/sensor.db,sha256=AVtiydrdtwAz2EFurO2Ult9SSRtre3r0akOBbL98LT0,554
39
- ophyd_async/epics/motion/__init__.py,sha256=tnmVRIwKa9PdN_xonJdAUD04UpEceh-hoD7XI62yDB0,46
40
- ophyd_async/epics/motion/motor.py,sha256=lQcA3PSPxA6XOnk8y5VuJGnKTq2VjkcO6ldCrssGr3M,3392
41
- ophyd_async/epics/signal/__init__.py,sha256=M1dz74L_SM5Qf3GRuQVKn-1Pvs-CEOCkoxyg1J-etHM,232
42
- ophyd_async/epics/signal/_epics_transport.py,sha256=DEIL0iYUAWssysVEgWGu1fHSM1l-ATV2kjUgPtDN9LY,858
43
- ophyd_async/epics/signal/pvi_get.py,sha256=YTNQ9nXi1v8Rig54goc4_nFTxoH_wmF3yl75FzkdXy4,479
44
- ophyd_async/epics/signal/signal.py,sha256=7GnGa4CgFgTwyHeh4wYOJ2GEWwdXsC6vHD3z9LsaHaM,2543
45
- ophyd_async/panda/__init__.py,sha256=1fyb-sP56K5zDmhKrn97YzMe7hL7LscHGgogqkGEobU,420
46
- ophyd_async/panda/panda.py,sha256=ThNzYarMlmyLqrUsDo-vjd-IxYLxnqKDIwvcBKWzED0,9901
47
- ophyd_async/panda/table.py,sha256=dLoRP4zYNOkD_s0Vkp2wVYAwkjVG8nNdf8-FaXOTfPo,5655
48
- ophyd_async-0.2.0.dist-info/LICENSE,sha256=pU5shZcsvWgz701EbT7yjFZ8rMvZcWgRH54CRt8ld_c,1517
49
- ophyd_async-0.2.0.dist-info/METADATA,sha256=hF1Gj4qjAmUc2eeaX3RA8awW41LYrkSieBJ5y36pqsI,7197
50
- ophyd_async-0.2.0.dist-info/WHEEL,sha256=Xo9-1PvkuimrydujYJAjF7pCkriuXBpUPEjma1nZyJ0,92
51
- ophyd_async-0.2.0.dist-info/entry_points.txt,sha256=O0YNJTEufO0w9BozXi-JurTy2U1_o0ypeCgJLQ727Jk,58
52
- ophyd_async-0.2.0.dist-info/top_level.txt,sha256=-hjorMsv5Rmjo3qrgqhjpal1N6kW5vMxZO3lD4iEaXs,12
53
- ophyd_async-0.2.0.dist-info/RECORD,,
File without changes