This is a Rust implementation of the standard ASCOM Alpaca API for astronomy devices.
It implements main Alpaca API clients and servers, as well as transparent support for auto-discovery mechanism and ImageBytes
encoding for camera images.
This crate defines two sets of compilation features that help to keep binary size & compilation speed in check by opting into only the features you need.
First set is along the client-server axis:
client
: Enables client-side access to Alpaca-capable devices.server
: Allows to expose your own devices as Alpaca servers.The second set of features is based on the device type and enables the corresponding trait:
all-devices
: Enables all of the below. Not recommended unless you're building a universal astronomy application.camera
: Enables support for cameras via the Camera
trait.covercalibrator
: Enables [...] the CoverCalibrator
trait.dome
: Enables Dome
.filterwheel
: Enables FilterWheel
.focuser
: Enables Focuser
.observingconditions
: Enables ObservingConditions
.rotator
: Enables Rotator
.switch
: Enables Switch
.telescope
: Enables Telescope
.Once you decided on the features you need, you can add this crate to your Cargo.toml
. For example, if I'm implementing an Alpaca camera driver, I'd add the following to my Cargo.toml
:
toml
[dependencies]
ascom-alpaca = { version = "0.1", features = ["client", "camera"] }
All the device type trait methods are async and correspond to the ASCOM Alpaca API. They all return ASCOMResult<...>
.
The Device
supertrait includes "ASCOM Methods Common To All Devices" from the Alpaca API, as well as a few custom metadata methods used for the device registration:
fn static_name(&self) -> &str
: Returns the static device name.fn unique_id(&self) -> &str
: Returns globally-unique device ID.Since async traits are not yet natively supported on stable Rust, the traits are implemented using the async-trait crate. Other than that, you should implement trait with all the Alpaca methods as usual:
```rust use ascomalpaca::ASCOMResult; use ascomalpaca::api::{Device, Camera}; use asynctrait::asynctrait;
struct MyCamera { // ... }
impl Device for MyCamera { fn static_name(&self) -> &str { "My Camera" }
fn unique_id(&self) -> &str {
"insert GUID here"
}
// ...
}
impl Camera for MyCamera {
async fn bayeroffsetx(&self) -> ASCOMResult
async fn bayer_offset_y(&self) -> ASCOMResult<i32> {
Ok(0)
}
// ...
} ```
Any skipped methods will default to the following values:
can_*
feature detection methods - to false
.Device::name
- to the result of Device::static_name()
.Device::interface_version
- to 3
(latest ASCOM interface version implemented by this crate).Device::supported_actions
- to an empty list.Err(ASCOMError::NOT_IMPLEMENTED)
. It's your responsibility to consult documentation and implement mandatory methods.Once traits are implemented, you can create a server, register your device(s), and start listening:
```rust use ascomalpaca::Server; use ascomalpaca::api::CargoServerInfo; use std::convert::Infallible;
// ...implement MyCamera...
async fn main() -> eyre::Result
// By default, the server will listen on dual-stack (IPv4 + IPv6) unspecified address with a randomly assigned port.
// You can change that by modifying the `listen_addr` field:
server.listen_addr.set_port(8000);
// Create and register your device(s).
server.devices.register(MyCamera { /* ... */ });
// Start the infinite server loop.
server.start().await
} ```
This will start both the main Alpaca server as well as an auto-discovery responder.
Examples:
examples/camera-server.rs
:
A cross-platform example exposing your connected webcam(s) as Alpaca Camera
s.
```log
env RUSTLOG=debug { cargo run --example camera-server --release } Finished release [optimized] target(s) in 0.60s Running
target\release\examples\camera-server.exe
2023-05-27T15:21:43.336191Z DEBUG cameraserver: Registering webcam webcam=Webcam { uniqueid: "150ddacb-7ad9-4754-b289-ae56210693e8::0", name: "Integrated Camera", description: "MediaFoundation Camera", maxformat: CameraFormat { resolution: Resolution { widthx: 1280, heighty: 720 }, format: MJPEG, framerate: 30 }, subframe: RwLock { data: Subframe { bin: Size { x: 1, y: 1 }, offset: Point { x: 0, y: 0 }, size: Size { x: 1280, y: 720 } } }, lastexposurestarttime: RwLock { data: None }, lastexposureduration: RwLock { data: None }, validbins: [1, 2, 4] } 2023-05-27T15:21:43.339433Z DEBUG ascomalpaca::server: Binding Alpaca server addr=[::]:8000 2023-05-27T15:21:43.342897Z INFO ascomalpaca::server: Bound Alpaca server boundaddr=[::]:8000 2023-05-27T15:21:43.369040Z WARN joinmulticastgroups{listenaddr=::}: ascomalpaca::server::discovery: err=An unknown, invalid, or unsupported option or level was specified in a getsockopt or setsockopt call. (os error 10042) 2023-05-27T15:21:43.370932Z DEBUG joinmulticastgroups{listenaddr=::}: ascomalpaca::server::discovery: return=() 2023-05-27T15:21:43.371861Z DEBUG ascom_alpaca::server: Bound Alpaca discovery server ```
Binning is implemented by switching the webcam to other supported resolutions which are proportional to the original.
Long exposures are simulated by stacking up individual frames up to the total duration. This approach can't provide precise requested exposure, but works well enough otherwise.
star-adventurer-alpaca
:
A fork of jsorrell/star-adventurer-alpaca
which implements the Alpaca API for the Star Adventurer mount over serial port.
The original project has pretty extensive functionality and used manual implementation of the Alpaca API, so it was a good test case for porting to this library.
If you know address of the device server you want to access, you can access it directly via Client
struct:
```rust use ascom_alpaca::Client;
let client = Client::new("http://localhost:8000")?;
// get_server_info
returns high-level metadata of the server.
println!("Server info: {:#?}", client.getserverinfo().await?);
// get_devices
returns an iterator over all the devices registered on the server.
// Each is represented as a TypedDevice
tagged enum encompassing all the device types as corresponding trait objects.
// You can either match on them to select the devices you're interested in, or, say, just print all of them:
println!("Devices: {:#?}", client.get_devices().await?.collect::
If you want to discover device servers on the local network, you can do that via the discovery::DiscoveryClient
struct:
```rust use ascomalpaca::discovery::DiscoveryClient; use ascomalpaca::Client; use futures::prelude::*;
// This holds configuration for the discovery client.
// You can customize prior to binding if you want.
let discoveryclient = DiscoveryClient::new();
// This results in a discovery client bound to a local socket.
// It's intentionally split out into a separate API step to encourage reuse,
// for example so that user could click "Refresh devices" button in the UI
// and the application wouldn't have to re-bind the socket every time.
let mut boundclient = discoveryclient.bind().await?;
// Now you can discover devices on the local networks.
boundclient.discoveraddrs()
// create a Client
for each discovered address
.map(|addr| Ok(Client::newfromaddr(addr)))
.tryfor_each(|client| async move {
/* ...do something with devices via each client... */
Ok::<_, eyre::Error>(())
})
.await?;
```
Or, if you just want to list all available devices and don't care about per-server information or errors:
rust
bound_client.discover_devices()
.for_each(|device| async move {
/* ...do something with each device... */
})
.await;
Keep in mind that discovery is a UDP-based protocol, so it's not guaranteed to be reliable.
Also, same device server can be discovered multiple times if it's available on multiple network interfaces.
While it's not possible to reliably deduplicate servers, you can deduplicate devices by storing them in something like HashSet
or in the same Devices
struct that is used for registering arbitrary devices on the server:
```rust use ascomalpaca::api::{Camera, Devices}; use ascomalpaca::discovery::DiscoveryClient; use ascom_alpaca::Client; use futures::prelude::*;
let devices =
DiscoveryClient::new()
.bind()
.await?
.discover_devices()
.collect::
// Now you can iterate over all the discovered devices via iter_all
:
for (typeddevice, indexwithincategory) in devices.iterall() {
println!("Discovered device: {typeddevice:#?} (index: {indexwithin_category})");
}
// ...or over devices in a specific category via iter<dyn Trait>
:
for camera in devices.iter::
Examples:
examples/discover.rs
:
A simple discovery example listing all the found servers and devices.
examples/camera-client.rs
:
A cross-platform GUI example showing a live preview stream from discovered Alpaca cameras.
Includes support for colour, monochrome and Bayer sensors with automatic colour conversion for the preview.
This crate uses tracing
framework for logging spans and events, integrating with the Alpaca ClientID
, ClientTransactionID
and ServerTransactionID
fields.
You can enable logging in your app by using any of the subscriber crates.
For example, tracing_subscriber::fmt
will log all the events to stderr depending on the RUST_LOG
environment variable:
rust
tracing_subscriber::fmt::init();
Since this is a library for communicating to networked devices, it should be tested against real devices at a higher level.
In particular, if you're implementing an Alpaca device, make sure to run ConformU - ASCOM's official conformance checker - against your device server.
Licensed under either of