#system #control #tango #distributed #watching

app rstango-top

Small text user-interface for watching activity in the Tango distributed control system

2 unstable releases

Uses new Rust 2024

0.2.0 Apr 15, 2026
0.1.0 Apr 15, 2026

#1344 in Text processing

Apache-2.0

470KB
9K SLoC

rsTango

Experimental project to create a new implementation of the Tango distributed control system, from scratch, and benefiting from core Rust features, such as ownership, lifetimes and its extensive generic type system.

This project is currently a personal project and is not affiliated with ESRF or the Tango Controls community.

Here you can find the core specification of the exposed API below.

Specification

Contrary to the current cppTango (and thus pyTango) implementation, the main goal is to abstract away the actual backend being used, namely omniORB for the RPC, and ZeroMQ for event system (which itself depends on omniORB).

/// root module, note that here we use a invalid rust syntax, just to define the interfaces
mod tango {

  /// core support for device proxies
  mod dev {

    /// proxy connection to a device
    struct DevProxy {

      /// returns a new proxy given a TRL, if the TRL is valid (e.g. not targeting an attribute)
      fn try_new(trl: impl IntoTrl) -> Option<Self>
      /// same as `try_new` but unwraps
      fn new(trl: impl IntoTrl) -> Self
      /// cheap clone
      impl Clone
      /// test equality of the target
      impl Eq
      /// hash the target
      impl Hash

      /// returns the proxy for the database used to connect to the device
      fn database(&mut self) -> Result<&mut DatabaseProxy>
      /// resolves the fully-qualified TRL of this device
      async fn trl(&mut self) -> Result<Trl>
      /// resolves the fully-qualified device name (domain/family/member)
      async fn name(&mut self) -> Result<String>
      /// resolves the admin device and returns a proxy to it
      async fn admin_dev(&mut self) -> Result<DevProxy>

      /// ping the device
      async fn ping(&mut self) -> Result<()>
      /// request the device state
      async fn state(&mut self) -> Result<DevState>
      /// request the device status
      async fn status(&mut self) -> Result<String>

      /// request the black box
      async fn black_box(&mut self, max: Option<usize>) -> Result<Vec<String>>

      /// execute a command and returns a handle to its output, to be interpreted
      async fn command_raw<'i, I: IntoData<'i>>(&mut self, name: &str, data: I) -> Result<CommandOutput>
      /// execute a command with generic input and output data
      async fn command<'i, I: IntoData<'i>, O: FromData<'_>>(&mut self, name: &str, data: I) -> Result<O>

      /// read all attributes and return a handle to interpret them
      async fn read_all_attr(&mut self) -> Result<ReadManyAttr>
      /// read many attributes and return a handle to interpret them
      async fn read_many_attr<Names: AsNames>(&mut self, names: Names) -> Result<ReadManyAttr>
      /// read one or more attributes given a spec type and their names:
      /// - with a spec type `T` and a single name `&'name str`, it returns a `AttrRead<'name, T>`
      /// - with a spec type `T` and an array of names `[&'name str; N]`, it returns a `[AttrRead<'name, T>; N]`
      /// - with a spec type `(A, B, ...)` and a tuple of names of the same size, it returns a `(AttrRead<'name, A>, AttrRead<'name, B>, ...)`
      async fn read_attr<Spec: ReadAttrSpec<Names>, Names: AsNames>(&mut self, names: Names) -> Result<Spec::Output>
      /// returns a builder for pushing many `AttrWrite<'name, ?>`
      fn write_attr_builder<'data>(&mut self) -> WriteAttrBuilder<'_, 'data>
      /// write one or more attributes given a spec of attributes:
      /// - with a single `AttrWrite<'name, T>` value;
      /// - with a single `(&'name str, T)` tuple, the `AttrWrite<'name, T>` is constructed on the fly with Valid quality and current time
      /// - some type that implements `IntoIterator` of any of these types
      /// - a tuple of any of these types
      async fn write_attr<Spec: WriteAttrSpec<'_, '_>>(&mut self, attrs: Spec) -> Result<WriteManyAttr>

      /// get information about all commands
      async fn get_all_command_info(&mut self) -> Result<GetManyCommandInfo>
      /// get command information about some specific commands
      async fn get_many_command_info<Names: AsNames>(&mut self, names: Names) -> Result<GetManyCommandInfo>

      /// read all attributes info
      async fn get_all_attr_info(&mut self) -> Result<GetManyAttrInfo>
      /// read many attributes info
      async fn get_many_attr_info<Names: AsNames>(&mut self, names: Names) -> Result<GetManyAttrInfo>
      // /// TODO:
      // fn set_many_attr_config<'dev, 'name>(&mut self,) -> Result<()>
      // /// TODO:
      // async fn set_attr_config<Spec: SetAttrInfoSpec<'_>>(&mut self, info: Spec) -> Result<()>

      /// subscribe to value events of given type on an attribute
      async fn subscribe_attr_value<T: FromData<'_>>(&mut self, name: &str, event: AttrValueEvent) -> Result<AttrChannel<T>>
      /// subscribe to read event on an attribute
      async fn subscribe_attr_ready(&mut self, name: &str) -> Result<AttrReadyChannel>

      /// list all properties of this device
      async fn list_prop(&mut self) -> Result<Vec<String>>
      /// get all properties for this device
      async fn get_all_prop(&mut self) -> Result<GetManyProp>
      /// get many properties from their names
      async fn get_many_prop<Names: AsNames>(&mut self, names: Names) -> Result<GetManyProp>
      /// forwarding to `DatabaseProxy::get_dev_prop`
      async fn get_prop<Spec: GetPropSpec<Names>, Names: AsNames>(&mut self, names: Names) -> Result<Spec::Output>
      /// create a builder for setting many properties at once on this device
      fn set_prop_builder<'data>(&mut self) -> SetPropBuilder<'_, 'data>
      /// forwarding to `DatabaseProxy::set_dev_prop`
      async fn set_prop<Spec: SetPropSpec<'_>>(&mut self, props: Spec) -> Result<()>
      /// forwarding to `DatabaseProxy::del_dev_prop`
      async fn del_prop<Names: AsNames>(&mut self, names: Names) -> Result<()>
      
    }

    /// supporting the different ways of representing, attribute, command and various 
    /// names, by converting it to an orderer slice of strings
    trait AsNames<'name>: Sealed {
      type Slice: AsRef<[&'name str]>
      fn into_slice(self) -> Self::Slice
    }

    /// supporting `DevProxy::command_raw`
    struct CommandOutput {
      /// try to parse the command output using the given type
      fn get<'s, O: FromData<'s>>(&'s self) -> Result<O>
    }

    /// supporting `DevProxy::read_many_attr` and `read_all_attr`
    /// default type for Names is when we request all attributes
    struct ReadManyAttr {
      /// returns the number of attributes read
      fn len(&self) -> usize
      /// try to parse the attribute values for the given type
      fn get<'s, T: FromData<'s>>(&'s self, index: usize) -> Result<AttrRead<'s, T>>
    }
    /// supporting `DevProxy::read_attr`
    trait ReadAttrSpec<Names>: Sealed {
      type Output;
    }

    /// supporting `DevProxy::write_many_attr`
    struct WriteAttrBuilder<'dev, 'data> {
      /// returns the current number of attributes to be written
      fn len(&self) -> usize
      /// push a new attribute write value to this builder
      fn push<T: IntoData<'data>>(&mut self, attr: AttrWrite<'data, T>)
      /// write all values to the device, returning the result for each attribute
      async fn write(self) -> Result<WriteManyAttr>
    }
    /// supporting `WriteManyAttrBuilder::write` and `DevProxy::write_attr`
    struct WriteManyAttr {
      /// returns the number of attributes that have been written
      fn len(&self) -> usize
      /// check the result of the attribute at given index
      fn check(&self, index: usize) -> Result<()>
    }
    /// supporting `DevProxy::write_attr`
    trait WriteAttrSpec: Sealed { }

    /// supporting `DevProxy::get_many_attr_info`
    /// default type for Names is when we request all attributes
    struct GetManyAttrInfo {
      /// the number of attributes
      fn len(&self) -> usize
      /// the information for a given attribute
      fn get(&self, index: usize) -> Result<AttrInfo<'name>>
    }

    /// supporting `DevProxy::get_many_command_info`
    /// default type for Names is when we request all attributes
    struct GetManyCommandInfo {
      /// the number of attributes
      fn len(&self) -> usize
      /// the information for a given attribute
      fn get(&self, index: usize) -> Result<CommandInfo<'name>>
    }

    /// failure stack for device
    struct DevFailed {
      /// create a new empty failure stack
      const fn new() -> Self
      fn push_frame(&mut self, frame: DevFailFrame)
      fn with_frame(mut self, frame: DevFailFrame) -> Self
      /// last frame refers to the origin of the failure propagation
      fn frames(&self) -> &[DevFailFrame]
      /// constructing the failed stack in order
      impl FromIterator<DevFailFrame>
    }

    struct DevFailFrame {
      reason: String
      severity: DevFailSeverity
      description: String
      origin: String
    }

    enum DevFailSeverity {
      Warning
      Error
      Panic
    }
    
    struct CommandInfo<'name> {
      /// The name of the read attribute.
      name: Cow<'name, str>
      /// The display level of the attribute.
      display_level: DisplayLevel
      /// Input data kind.
      in_kind: DataKind
      /// Output data kind.
      out_kind: DataKind
      /// Description for the output data.
      in_description: Option<String>
      /// Description for the input data.
      out_description: Option<String>
    }

    struct AttrRead<'name, T> {
      name: Cow<'name, str>
      value: Option<T>
      w_value: Option<T>
      quality: AttrQuality
      time: SystemTime
    }

    struct AttrWrite<'name, T> {
      name: Cow<'name, str>
      value: Option<T>
      quality: AttrQuality
      time: SystemTime
    }

    enum AttrQuality {
      Valid
      Invalid
      Alarm
      Changing
      Warning
    }

    struct AttrInfo<'name> {
      config: AttrConfig<'name>
      mode: AttrMode<'name>
      data_kind: DataKind
    }

    enum AttrMode<'name> {
      Read
      ReadWithWrite {
        write_attr_name: Cow<'name, str>
      }
      Write
      ReadWrite
    }

    struct AttrConfig<'name> {
      name: Cow<'name, str>
      description: Option<String>
      label: Option<String>
    }

    /// all the device states
    enum DevState {
      On
      Off
      Close
      Open
      Insert
      Extract
      Moving
      Standby
      Fault
      Init
      Running
      Alarm
      Disable
      Unknown
    }

    enum DevSource {
      Dev
      Cache
      CacheDev
    }

    enum DisplayLevel {
      Operator
      Expert
    }

    enum AttrValueEvent {
      Change
      Periodic
      Archive
      User
      Alarm
    }

    struct AttrChannel<T> {
      /// indefinitely wait for an attribute to have a new value
      async fn wait<T: FromData<'_>>(&mut self) -> Result<AttrRead<'_, T>>
    }

    struct AttrReadyChannel {
      /// indefinitely wait for an attribute to be ready
      async fn wait(&mut self) -> Result<&'_ str>
    }

  }
  /// wrapper around device proxy but with standard interface for databases.
  mod database {

    /// a wrapper around a DevProxy that is known to be a database
    struct DatabaseProxy {

      /// new proxy to the environment database(s) (TANGO_HOST)
      fn new_from_env() -> Self
      /// new proxy to a specific data
      fn new(host: &str, port: u16) -> Self

      /// request a device for import on the database (DbImportDevice)
      async fn import_dev(&mut self, dev_name: &str) -> Result<Option<ImportDevInfo>>

      /// list all free properties given a wildcard (DbGetPropertyList)
      async fn list_free_prop(&mut self, obj_name: &str, wildcard: &str) -> Result<Vec<String>>
      /// list all class properties given a wildcard (DbGetClassPropertyList/DbGetClassPropertyListWildcard)
      async fn list_class_prop(&mut self, class_name: &str, wildcard: &str) -> Result<Vec<String>>
      /// list all device properties given a wildcard (DbGetDevicePropertyList)
      async fn list_dev_prop(&mut self, dev_name: &str, wildcard: &str) -> Result<Vec<String>>

      /// return a handle for reading all properties of an object
      async fn get_all_free_prop(&mut self, obj_name: &str) -> Result<GetManyProp>
      /// return a handle for reading all properties of a class
      async fn get_all_class_prop(&mut self, class_name: &str) -> Result<GetManyProp>
      /// return a handle for reading all properties of a device
      async fn get_all_dev_prop(&mut self, dev_name: &str) -> Result<GetManyProp>

      /// return a handle for reading many properties of an object (DbGetProperty)
      async fn get_many_free_prop<Names: AsNames>(&mut self, obj_name: &str, names: Names) -> Result<GetManyProp>
      /// return a handle for reading many properties of a class (DbGetClassProperty)
      async fn get_many_class_prop<Names: AsNames>(&mut self, class_name: &str, names: Names) -> Result<GetManyProp>
      /// return a handle for reading many properties of a device (DbGetDeviceProperty)
      async fn get_many_dev_prop<Names: AsNames>(&mut self, dev_name: &str, names: Names) -> Result<GetManyProp>

      /// get one or multiple object properties
      async fn get_free_prop<Spec: GetPropSpec<Names>, Names: AsNames>(&mut self, obj_name: &str, names: Names) -> Result<Spec::Output>
      /// get one or multiple class properties
      async fn get_class_prop<Spec: GetPropSpec<Names>, Names: AsNames>(&mut self, class_name: &str, names: Names) -> Result<Spec::Output>
      /// get one or multiple device properties
      async fn get_dev_prop<Spec: GetPropSpec<Names>, Names: AsNames>(&mut self, dev_name: &str, names: Names) -> Result<Spec::Output>
      
      /// returns a new builder for setting one or multiple object properties (DbPutProperty)
      async fn set_free_prop_builder<'data>(&mut self, obj_name: &'data str) -> SetPropBuilder<'_, 'data>
      /// returns a new builder for setting one or multiple class properties (DbPutClassProperty)
      async fn set_class_prop_builder<'data>(&mut self, class_name: &'data str) -> SetPropBuilder<'_, 'data>
      /// returns a new builder for setting one or multiple device properties (DbPutDeviceProperty)
      async fn set_dev_prop_builder<'data>(&mut self, dev_name: &'data str) -> SetPropBuilder<'_, 'data>

      /// set one or multiple object properties
      async fn set_free_prop<'data, Spec: SetPropSpec<'data>>(&mut self, obj_name: &'data str, props: Spec) -> Result<()>
      /// set one or multiple class properties
      async fn set_class_prop<'data, Spec: SetPropSpec<'data>>(&mut self, class_name: &'data str, props: Spec) -> Result<()>
      /// set one or multiple device properties
      async fn set_dev_prop<'data, Spec: SetPropSpec<'data>>(&mut self, dev_name: &'data str, props: Spec) -> Result<()>

      /// delete many device properties (DbDeleteProperty)
      async fn del_free_prop<Names: AsNames>(&mut self, obj_name: &str, names: Names) -> Result<()>
      /// delete many device properties (DbDeleteClassProperty)
      async fn del_class_prop<Names: AsNames>(&mut self, class_name: &str, names: Names) -> Result<()>
      /// delete many device properties (DbDeleteDeviceProperty)
      async fn del_dev_prop<Names: AsNames>(&mut self, dev_name: &str, names: Names) -> Result<()>

    }

    struct ImportDevInfo {
      exported: bool
      pid: i32
      name: String
      itr: String
      version: String
      server: String
      host: String
      class: Option<String>
    }

    struct PropValue<'name, T> {
      name: Cow<'name, T>
      value: T
    }

    struct GetManyProp<'name, Names: AsNames<'name> = ()> {
      
      fn len(&self) -> usize
      fn get<'d, T: FromProp<'d>>(&'d self, index: usize) -> Result<PropValue<'name, T>>
      
      impl Debug

    }

  }

  /// parsing of TRL
  mod trl {

    /// represents a validated Tango Resource Locator
    struct Trl {
      /// returns the string representation
      fn as_str(&self) -> &str
      /// if present, returns host/port
      fn host_and_port(&self) -> Option<(&str, u16)>
      /// returns the pointed object
      fn object(&self) -> TrlObject<'_>
      /// if present, returns the property pointed on the object
      fn property(&self) -> Option<&str>
      /// returns false if database should not be used (#dbase=no)
      fn db(&self) -> bool
      /// for parsing the TRL from any string
      impl FromStr<Err = ParseTrlError>
    }

    /// represent the different kind of object pointed by a TRL
    enum TrlObject<'a> {
      /// an alias for a device or an attribute
      Alias(&'a str)
      /// a device name
      Dev(&'a str)
      /// a device name + attribute name
      Attr(&'a str, &'a str)
    }

    /// for types that can be parsed into a TRL
    trait IntoTrl {
        fn into_trl(self) -> Result<Trl, ParseTrlError>
        /// returns itself
        impl for Trl
        /// forward to FromStr
        impl for &'_ str
    }

    /// parsing error
    enum ParseTrlError {
      InvalidProtocol
      InvalidHost
      InvalidPort
      InvalidHash
      InvalidPath
      InvalidDeviceName
      InvalidAttrName
      InvalidAlias
      InvalidPropertyName
    }

  }

  /// general error handling for Tango
  mod error {

    /// result type alias
    type Result<T> = std::result::Result<T, Error>

    /// the common error type
    enum Error {
      NotHandled
      DatabaseNotConnected
      DatabaseNeeded
      DevNotFound
      ServerNotConnected
      DevNotConnected
      CommandNameInvalid
      CommandNotFound
      AttrNameInvalid
      AttrNotFound
      AttrNotWritable
      EventNotConnected
      InvalidIndex
      Data(Box<str>)
      TimedOut
      Failed(DevFailed)
      Internal(#[source] Box<dyn Error + Send + Sync>)
    }

  }

}

Dependencies

~20–28MB
~416K SLoC