Using Devices#
In this tutorial we will create a bluesky RunEngine, instantiate some existing ophyd-async Devices, and use them in some bluesky plans. It assumes you have already run through the Bluesky tutorial on The RunEngine.
Run the demo#
Ophyd-async ships with some simulated devices and a demo script that will create them along with a RunEngine. Let’s take a look at it now:
"""Used for tutorial `Using Devices`."""
# Import bluesky and ophyd
from tempfile import mkdtemp
import bluesky.plan_stubs as bps # noqa: F401
import bluesky.plans as bp # noqa: F401
from bluesky.callbacks.best_effort import BestEffortCallback
from bluesky.run_engine import RunEngine, autoawait_in_bluesky_event_loop
from ophyd_async import sim
from ophyd_async.core import StaticPathProvider, UUIDFilenameProvider, init_devices
# Create a run engine and make ipython use it for `await` commands
RE = RunEngine(call_returns_result=True)
autoawait_in_bluesky_event_loop()
# Add a callback for plotting
bec = BestEffortCallback()
RE.subscribe(bec)
# Make a pattern generator that uses the motor positions
# to make a test pattern. This simulates the real life process
# of X-ray scattering off a sample
pattern_generator = sim.PatternGenerator()
# Make a path provider that makes UUID filenames within a static
# temporary directory
path_provider = StaticPathProvider(UUIDFilenameProvider(), mkdtemp())
# All Devices created within this block will be
# connected and named at the end of the with block
with init_devices():
# Create a sample stage with X and Y motors that report their positions
# to the pattern generator
stage = sim.SimStage(pattern_generator)
# Make a detector device that gives the point value of the pattern generator
# when triggered
pdet = sim.SimPointDetector(pattern_generator)
# Make a detector device that gives a gaussian blob with intensity based
# on the point value of the pattern generator when triggered
bdet = sim.SimBlobDetector(path_provider, pattern_generator)
We will explain the contents in more detail later on, but for now let’s run it in an interactive ipython shell:
$ ipython -i -m ophyd_async.sim
Python 3.11.11 (main, Dec 4 2024, 20:38:25) [GCC 12.2.0]
Type 'copyright', 'credits' or 'license' for more information
IPython 8.30.0 -- An enhanced Interactive Python. Type '?' for help.
In [1]:
This has launched an ipython shell, told it to import and run the demo script packaged inside ophyd_async.sim
, then return to an interactive prompt.
Investigate the Devices#
We will look at the stage.x
and y
motors first. If we examine them we can see that they have a name:
In [1]: stage.x.name
Out[1]: 'stage-x'
But if we try to call any of the other methods like read()
we will see that it doesn’t return the value, but a coroutine:
In [2]: stage.x.read()
Out[2]: <coroutine object StandardReadable.read at 0x7f9c5c105220>
This is because ophyd-async devices implement async versions of the bluesky verbs. To get the value we can await
it:
In [3]: await stage.x.read()
Out[3]:
{'x-user_readback': {'value': 0.0,
'timestamp': 367727.615860209,
'alarm_severity': 0}}
Run some plans#
Although it is useful to run the verbs using the await
syntax for debugging, most of the time we will run them via plans executed by the bluesky.run_engine.RunEngine
. For instance we can read it using the bps.rd
plan stub:
In [4]: RE(bps.rd(stage.x))
Out[4]: RunEngineResult(run_start_uids=(), plan_result=0.0, exit_status='success', interrupted=False, reason='', exception=None)
and move it using the bps.mv
plan sub:
In [5]: RE(bps.mv(stage.x, 1.5))
Out[5]: RunEngineResult(run_start_uids=(), plan_result=(<WatchableAsyncStatus, device: x, task: <coroutine object WatchableAsyncStatus._notify_watchers_from at 0x7f9c71791940>, done>,), exit_status='success', interrupted=False, reason='', exception=None)
In [6]: RE(bps.rd(stage.x))
Out[6]: RunEngineResult(run_start_uids=(), plan_result=1.5, exit_status='success', interrupted=False, reason='', exception=None)
Run a grid scan#
There is also a point detector that changes its 3 channels of output based on the positions of the stage.x
and stage.y
motors, so we can use it in a bp.grid_scan
:
In [1]: RE(bp.grid_scan([pdet], stage.x, 1, 2, 3, stage.y, 2, 3, 3))
Transient Scan ID: 1 Time: 2025-02-20 14:46:38
Persistent Unique Scan ID: 'ae2393b4-6598-4f2e-a10c-ed9dcc594b03'
Trigger took 0.10090422630310059
New stream: 'primary'
+-----------+------------+------------+------------+----------------------+----------------------+----------------------+
| seq_num | time | stage-x | stage-y | pdet-channel-1-value | pdet-channel-2-value | pdet-channel-3-value |
+-----------+------------+------------+------------+----------------------+----------------------+----------------------+
| 1 | 14:46:38.9 | 1.000 | 2.000 | 921 | 887 | 859 |
Trigger took 0.10077595710754395
| 2 | 14:46:39.9 | 1.000 | 2.500 | 959 | 926 | 898 |
Trigger took 0.10067224502563477
| 3 | 14:46:40.8 | 1.000 | 3.000 | 937 | 903 | 875 |
Trigger took 0.10073423385620117
| 4 | 14:46:41.7 | 1.500 | 2.000 | 976 | 975 | 974 |
Trigger took 0.10071492195129395
| 5 | 14:46:42.6 | 1.500 | 2.500 | 843 | 843 | 842 |
Trigger took 0.10068893432617188
| 6 | 14:46:43.5 | 1.500 | 3.000 | 660 | 660 | 659 |
Trigger took 0.10079550743103027
| 7 | 14:46:44.4 | 2.000 | 2.000 | 761 | 740 | 722 |
Trigger took 0.10068082809448242
| 8 | 14:46:45.2 | 2.000 | 2.500 | 537 | 516 | 498 |
Trigger took 0.10111546516418457
| 9 | 14:46:46.1 | 2.000 | 3.000 | 487 | 467 | 448 |
+-----------+------------+------------+------------+----------------------+----------------------+----------------------+
generator grid_scan ['ae2393b4'] (scan num: 1)
Out[1]: RunEngineResult(run_start_uids=('ae2393b4-6598-4f2e-a10c-ed9dcc594b03',), plan_result='ae2393b4-6598-4f2e-a10c-ed9dcc594b03', exit_status='success', interrupted=False, reason='', exception=None)
data:image/s3,"s3://crabby-images/0d5f7/0d5f71908590585644b88bbf414d676032c33586" alt="../_images/sim_grid_scan.png"
This detector produces a single point of information for each channel at each motor value. This means that the Best-Effort Callback is able to print a tabular form of the scan.
There is also a blob detector that produces a gaussian blob with intensity based on the positions of the stage.x
and stage.y
motors, writing the data to an HDF file. You can also use this in a grid scan, but there will be no data displayed as the BestEffortCallback
doesn’t know how to read data from file:
In [2]: RE(bp.grid_scan([bdet], stage.x, 1, 2, 3, stage.y, 2, 3, 3))
Transient Scan ID: 2 Time: 2025-02-20 14:46:47
Persistent Unique Scan ID: '74ea2f30-7599-48d0-bdb6-9a0db3cce07f'
New stream: 'primary'
+-----------+------------+------------+------------+------------+
| seq_num | time | stage-x | stage-y | bdet-sum |
+-----------+------------+------------+------------+------------+
| 1 | 14:46:47.5 | 1.000 | 2.000 | |
| 2 | 14:46:47.6 | 1.000 | 2.500 | |
| 3 | 14:46:47.8 | 1.000 | 3.000 | |
| 4 | 14:46:47.9 | 1.500 | 2.000 | |
| 5 | 14:46:48.1 | 1.500 | 2.500 | |
| 6 | 14:46:48.2 | 1.500 | 3.000 | |
| 7 | 14:46:48.4 | 2.000 | 2.000 | |
| 8 | 14:46:48.5 | 2.000 | 2.500 | |
| 9 | 14:46:48.6 | 2.000 | 3.000 | |
+-----------+------------+------------+------------+------------+
generator grid_scan ['74ea2f30'] (scan num: 2)
Out[2]: RunEngineResult(run_start_uids=('74ea2f30-7599-48d0-bdb6-9a0db3cce07f',), plan_result='74ea2f30-7599-48d0-bdb6-9a0db3cce07f', exit_status='success', interrupted=False, reason='', exception=None)
See also
A more interactive scanning tutorial including live plotting of the data from file is in the process of being written in the bluesky cookbook.
Examine the script#
We will now walk through the script section by section and examine what each part does. First of all we import the bluesky and ophyd libraries:
from tempfile import mkdtemp
import bluesky.plan_stubs as bps # noqa: F401
import bluesky.plans as bp # noqa: F401
from bluesky.callbacks.best_effort import BestEffortCallback
from bluesky.run_engine import RunEngine, autoawait_in_bluesky_event_loop
from ophyd_async import sim
from ophyd_async.core import StaticPathProvider, UUIDFilenameProvider, init_devices
After this we create a RunEngine:
RE = RunEngine(call_returns_result=True)
autoawait_in_bluesky_event_loop()
We pass call_returns_result=True
to the RunEngine so that we can see the result of bps.rd
above. We call autoawait_in_bluesky_event_loop()
so that when we await bps.rd(x)
it will happen in the same event loop that the RunEngine uses rather than an IPython specific one. This avoids some surprising behavior that occurs when devices are accessed from multiple event loops.
We then setup plotting of the resulting scans:
bec = BestEffortCallback()
RE.subscribe(bec)
This subscribes to the emitted bluesky Documents, and interprets them for plotting. In this case it made a table of points for the motors and each channel of the point detector, and plots of the point detector channels in a gridded pattern.
Just for the simulation we need something to produce the test data:
pattern_generator = sim.PatternGenerator()
This is passed to all the Devices so they can tell it the X and Y positions of the motors and get simulated point and gaussian blob data from it. In production you would pass around references to the control system (EPICS PV prefixes or Tango Resource Locations) instead of creating an object here. This is explored in more detail in Implementing Devices.
Next up is the path provider:
path_provider = StaticPathProvider(UUIDFilenameProvider(), mkdtemp())
This is how we specify in which location file-writing detectors store their data. In this example we choose to write to a static temporary directory using the StaticPathProvider
, and to name each file within it with a UUID using the UUIDFilenameProvider
. Other PathProviders allow this to be customized. In production we would chose a location on a filesystem that would be accessible by downstream consumers of the scan documents.
Finally we create and connect the Devices:
with init_devices():
# Create a sample stage with X and Y motors that report their positions
# to the pattern generator
stage = sim.SimStage(pattern_generator)
# Make a detector device that gives the point value of the pattern generator
# when triggered
pdet = sim.SimPointDetector(pattern_generator)
# Make a detector device that gives a gaussian blob with intensity based
# on the point value of the pattern generator when triggered
bdet = sim.SimBlobDetector(path_provider, pattern_generator)
The first thing to note is the with
statement. This uses a init_devices
as a context manager to collect up the top level Device
instances created in the context, and run the following:
If
set_name=True
(the default), then callDevice.set_name
passing the name of the variable within the context. For example, here we callpdet.set_name("pdet")
If
connect=True
(the default), then callDevice.connect
in parallel for all top level Devices, waiting for up totimeout
seconds. For example, here we will connectstage
,pdet
andbdet
at the same time. This parallel connect speeds up connection to the underlying control system.If
mock=True
is passed, then don’t connect to the control system, but set Devices into mock mode for testing.
Within it the device creation happens, in this case the stage
with x
and y
motors, and the two detectors.
Conclusion#
In this tutorial we have instantiated some existing ophyd-async devices, seen how they can be connected and named, and used them in some basic plans. Read on to see how to implement support for devices via a control system like EPICS or Tango.