diff --git a/StatsCrosslet-DS/StatsCrosslet.py b/StatsCrosslet-DS/StatsCrosslet.py new file mode 100644 index 0000000000000000000000000000000000000000..2e0bb6ebe0b8b811971c50d1acc04d17ab172564 --- /dev/null +++ b/StatsCrosslet-DS/StatsCrosslet.py @@ -0,0 +1,318 @@ +# -*- coding: utf-8 -*- +# +# This file is part of the StatsCrosslet project +# +# +# +# Distributed under the terms of the APACHE license. +# See LICENSE.txt for more info. + +""" OPC-UA client for LOFAR stations crosslet stats + +""" + +# PyTango imports +import tango +from tango import DebugIt +from tango.server import run +from tango.server import Device +from tango.server import attribute, command, pipe +from tango.server import device_property +from tango import AttrQuality, DispLevel, DevState +from tango import AttrWriteType, PipeWriteType +# Additional import +# PROTECTED REGION ID(StatsCrosslet.additionnal_import) ENABLED START # +import opcua +import numpy +import traceback +import threading +import time +# PROTECTED REGION END # // StatsCrosslet.additionnal_import + +__all__ = ["StatsCrosslet", "main"] + + +class StatsCrosslet(Device): + """ + + **Properties:** + + - Device Property + OPC_Server_Name + - Type:'DevString' + OPC_Server_Port + - Type:'DevULong' + OPC_time_out + - Type:'DevULong' + """ + # PROTECTED REGION ID(StatsCrosslet.class_variable) ENABLED START # + client = 0 + ns = 0 + obj = 0 + record_cross = 0 + data = [] + data_lock = threading.Lock() + stop_data_read_loop = threading.Event() + data_acquisition_is_active = threading.Event() + data_read_loop = threading.Thread() + + def read_data(self): + # This is the the thread that continuously reads the crosslet + # statistics from the station and feeds it into the Tango + # property system. + while self.stop_data_read_loop.is_set() is False: + time.sleep(self._pause_time) + if self.data_acquisition_is_active.is_set() is True: + data = [] + try: + timestamp, visibilities, rcu_modes = self.obj.call_method(self.record_cross, self._subband, self._integration_time) + self.data_lock.acquire() + self._time_stamp = timestamp + self._raw_visibilities = visibilities + self._visibilities = numpy.array(visibilities)[0] + 1j * numpy.array(visibilities[1]) + self._rcu_modes = rcu_modes + self.data_lock.release() + + except Exception as e: + print("Cannot call the method %s in the OPC-UA server %s. Trace: %s" % (self.record_cross, self.OPC_Server_Name, traceback.format_exc +())) + # PROTECTED REGION END # // StatsCrosslet.class_variable + + # ----------------- + # Device Properties + # ----------------- + + OPC_Server_Name = device_property( + dtype='DevString', + default_value="okeanos" + ) + + OPC_Server_Port = device_property( + dtype='DevULong', + default_value=55556 + ) + + OPC_time_out = device_property( + dtype='DevULong', + default_value=1000 + ) + + # ---------- + # Attributes + # ---------- + + subband = attribute( + dtype='DevUShort', + access=AttrWriteType.READ_WRITE, + ) + + integration_time = attribute( + dtype='DevDouble', + ) + + time_stamp = attribute( + dtype='DevDouble', + ) + + pause_time = attribute( + dtype='DevDouble', + access=AttrWriteType.READ_WRITE, + ) + + visibilities = attribute( + dtype=('DevDouble',), + max_dim_x=96, + ) + + rcu_modes = attribute( + dtype=('DevString',), + max_dim_x=96, + ) + + raw_visibilities = attribute( + dtype=(('DevDouble',),), + max_dim_x=96, max_dim_y=96, + ) + + # ----- + # Pipes + # ----- + + pipe_visibilities = pipe( + ) + pipe_raw_visibilities = pipe( + ) + + # --------------- + # General methods + # --------------- + + def init_device(self): + """Initialises the attributes and properties of the StatsCrosslet.""" + Device.init_device(self) + # PROTECTED REGION ID(StatsCrosslet.init_device) ENABLED START # + self._visibilities = (0.0,) + try: + self.client = opcua.Client("opc.tcp://{}:{}/".format(self.OPC_Server_Name, self.OPC_Server_Port), self.OPC_time_out) + self.client.connect() + ns = self.client.get_namespace_index("http://lofar.eu") + self.obj = self.client.get_root_node().get_child(["0:Objects", "{}:StationMetrics".format(ns), "{}:RCU".format(ns)]) + self.record_cross = "{}:record_cross".format(ns) + self.data_read_loop = threading.Thread(target = self.read_data) + self.data_acquisition_is_active.set() + self.stop_data_read_loop.clear() + self.data_read_loop.start() + except Exception as e: + print("Cannot connect to the OPC-UA server %s. Trace: %s" % (self.OPC_Server_Name, traceback.format_exc())) + # PROTECTED REGION END # // StatsCrosslet.init_device + + def always_executed_hook(self): + """Method always executed before any TANGO command is executed.""" + # PROTECTED REGION ID(StatsCrosslet.always_executed_hook) ENABLED START # + # PROTECTED REGION END # // StatsCrosslet.always_executed_hook + + def delete_device(self): + """Hook to delete resources allocated in init_device. + + This method allows for any memory or other resources allocated in the + init_device method to be released. This method is called by the device + destructor and by the device Init command. + """ + # PROTECTED REGION ID(StatsCrosslet.delete_device) ENABLED START # + self.data_acquisition_is_active.clear() + self.stop_data_read_loop.set() + self.data_read_loop.join() + if self.client is not Null: + self.client.close_session() + self.client_secure_channel() + # PROTECTED REGION END # // StatsCrosslet.delete_device + # ------------------ + # Attributes methods + # ------------------ + + def read_subband(self): + # PROTECTED REGION ID(StatsCrosslet.subband_read) ENABLED START # + """Return the subband attribute.""" + return self._subband + # PROTECTED REGION END # // StatsCrosslet.subband_read + + def write_subband(self, value): + # PROTECTED REGION ID(StatsCrosslet.subband_write) ENABLED START # + """Set the subband attribute.""" + self._subband = value + # PROTECTED REGION END # // StatsCrosslet.subband_write + + def read_integration_time(self): + # PROTECTED REGION ID(StatsCrosslet.integration_time_read) ENABLED START # + """Return the integration_time attribute.""" + return self._integration_time + # PROTECTED REGION END # // StatsCrosslet.integration_time_read + + def read_time_stamp(self): + # PROTECTED REGION ID(StatsCrosslet.time_stamp_read) ENABLED START # + """Return the time_stamp attribute.""" + self.data_lock.acquire() + time_stamp = self._time_stamp + self.data_lock.release() + return time_stamp + # PROTECTED REGION END # // StatsCrosslet.time_stamp_read + + def read_pause_time(self): + # PROTECTED REGION ID(StatsCrosslet.pause_time_read) ENABLED START # + """Return the pause_time attribute.""" + return self._pause_time + # PROTECTED REGION END # // StatsCrosslet.pause_time_read + + def write_pause_time(self, value): + # PROTECTED REGION ID(StatsCrosslet.pause_time_write) ENABLED START # + """Set the pause_time attribute.""" + self._pause_time = value + # PROTECTED REGION END # // StatsCrosslet.pause_time_write + + def read_visibilities(self): + # PROTECTED REGION ID(StatsCrosslet.visibilities_read) ENABLED START # + """Return the visibilities attribute.""" + self.data_lock.acquire() + visibilities = self._visibilities + self.data_lock.release() + return visibilities + # PROTECTED REGION END # // StatsCrosslet.visibilities_read + + def read_rcu_modes(self): + # PROTECTED REGION ID(StatsCrosslet.rcu_modes_read) ENABLED START # + """Return the rcu_modes attribute.""" + self.data_lock.acquire() + rcu_modes = self._rcu_modes + self.data_lock.release() + return rcu_modes + # PROTECTED REGION END # // StatsCrosslet.rcu_modes_read + + def read_raw_visibilities(self): + # PROTECTED REGION ID(StatsCrosslet.raw_visibilities_read) ENABLED START # + """Return the raw_visibilities attribute.""" + self.data_lock.acquire() + raw_visibilities = self._raw_visibilities + self.data_lock.release() + return raw_visibilities + # PROTECTED REGION END # // StatsCrosslet.raw_visibilities_read + + # ------------- + # Pipes methods + # ------------- + + def read_pipe_visibilities(self): + # PROTECTED REGION ID(StatsCrosslet.pipe_visibilities_read) ENABLED START # + return dict(x=0, y=0) + # PROTECTED REGION END # // StatsCrosslet.pipe_visibilities_read + + def read_pipe_raw_visibilities(self): + # PROTECTED REGION ID(StatsCrosslet.pipe_raw_visibilities_read) ENABLED START # + return dict(x=0, y=0) + # PROTECTED REGION END # // StatsCrosslet.pipe_raw_visibilities_read + + # -------- + # Commands + # -------- + + @command( + ) + @DebugIt() + def start_acquisition(self): + # PROTECTED REGION ID(StatsCrosslet.start_acquisition) ENABLED START # + """ + Start the data acquisition of the station`s crosslet stats. + + :param argin: 'DevULong' + + :return:None + """ + self.data_acquisition_is_active.set() + # PROTECTED REGION END # // StatsCrosslet.start_acquisition + + @command( + ) + @DebugIt() + def stop_acquisition(self): + # PROTECTED REGION ID(StatsCrosslet.stop_acquisition) ENABLED START # + """ + Stop the data acquisition. + + :return:None + """ + self.data_acquisition_is_active.clear() + # PROTECTED REGION END # // StatsCrosslet.stop_acquisition + +# ---------- +# Run server +# ---------- + + +def main(args=None, **kwargs): + """Main function of the StatsCrosslet module.""" + # PROTECTED REGION ID(StatsCrosslet.main) ENABLED START # + return run((StatsCrosslet,), args=args, **kwargs) + # PROTECTED REGION END # // StatsCrosslet.main + + +if __name__ == '__main__': + main() diff --git a/StatsCrosslet-DS/StatsCrosslet.xmi b/StatsCrosslet-DS/StatsCrosslet.xmi new file mode 100644 index 0000000000000000000000000000000000000000..884ebb33d7a932c094ea3b0dde400bb1f1742cc3 --- /dev/null +++ b/StatsCrosslet-DS/StatsCrosslet.xmi @@ -0,0 +1,119 @@ +<?xml version="1.0" encoding="ASCII"?> +<pogoDsl:PogoSystem xmi:version="2.0" xmlns:xmi="http://www.omg.org/XMI" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:pogoDsl="http://www.esrf.fr/tango/pogo/PogoDsl"> + <classes name="StatsCrosslet" pogoRevision="9.6"> + <description description="" title="OPC-UA client for LOFAR stations crosslet stats" sourcePath="/opt/tango/StatsCrosslet-DS" language="PythonHL" filestogenerate="XMI file,Code files,Protected Regions" license="APACHE" copyright="" hasMandatoryProperty="false" hasConcreteProperty="true" hasAbstractCommand="false" hasAbstractAttribute="false"> + <inheritances classname="Device_Impl" sourcePath=""/> + <identification contact="at astron.nl - jurges" author="jurges" emailDomain="astron.nl" classFamily="Acquisition" siteSpecific="" platform="Unix Like" bus="Ethernet" manufacturer="" reference=""/> + </description> + <deviceProperties name="OPC_Server_Name" description=""> + <type xsi:type="pogoDsl:StringType"/> + <status abstract="false" inherited="false" concrete="true" concreteHere="true"/> + <DefaultPropValue>okeanos</DefaultPropValue> + </deviceProperties> + <deviceProperties name="OPC_Server_Port" description=""> + <type xsi:type="pogoDsl:UIntType"/> + <status abstract="false" inherited="false" concrete="true" concreteHere="true"/> + <DefaultPropValue>55556</DefaultPropValue> + </deviceProperties> + <deviceProperties name="OPC_time_out" description=""> + <type xsi:type="pogoDsl:UIntType"/> + <status abstract="false" inherited="false" concrete="true" concreteHere="true"/> + <DefaultPropValue>1000</DefaultPropValue> + </deviceProperties> + <commands name="State" description="This command gets the device state (stored in its device_state data member) and returns it to the caller." execMethod="dev_state" displayLevel="OPERATOR" polledPeriod="0"> + <argin description="none"> + <type xsi:type="pogoDsl:VoidType"/> + </argin> + <argout description="Device state"> + <type xsi:type="pogoDsl:StateType"/> + </argout> + <status abstract="true" inherited="true" concrete="true"/> + </commands> + <commands name="Status" description="This command gets the device status (stored in its device_status data member) and returns it to the caller." execMethod="dev_status" displayLevel="OPERATOR" polledPeriod="0"> + <argin description="none"> + <type xsi:type="pogoDsl:VoidType"/> + </argin> + <argout description="Device status"> + <type xsi:type="pogoDsl:ConstStringType"/> + </argout> + <status abstract="true" inherited="true" concrete="true"/> + </commands> + <commands name="start_acquisition" description="Start the data acquisition of the station`s crosslet stats." execMethod="start_acquisition" displayLevel="OPERATOR" polledPeriod="0" isDynamic="false"> + <argin description=""> + <type xsi:type="pogoDsl:VoidType"/> + </argin> + <argout description=""> + <type xsi:type="pogoDsl:VoidType"/> + </argout> + <status abstract="false" inherited="false" concrete="true" concreteHere="true"/> + </commands> + <commands name="stop_acquisition" description="Stop the data acquisition." execMethod="stop_acquisition" displayLevel="OPERATOR" polledPeriod="0" isDynamic="false"> + <argin description=""> + <type xsi:type="pogoDsl:VoidType"/> + </argin> + <argout description=""> + <type xsi:type="pogoDsl:VoidType"/> + </argout> + <status abstract="false" inherited="false" concrete="true" concreteHere="true"/> + </commands> + <attributes name="subband" attType="Scalar" rwType="READ_WRITE" displayLevel="OPERATOR" polledPeriod="0" maxX="" maxY="" allocReadMember="true" isDynamic="false"> + <dataType xsi:type="pogoDsl:UShortType"/> + <changeEvent fire="false" libCheckCriteria="false"/> + <archiveEvent fire="false" libCheckCriteria="false"/> + <dataReadyEvent fire="false" libCheckCriteria="true"/> + <status abstract="false" inherited="false" concrete="true" concreteHere="true"/> + <properties description="" label="" unit="" standardUnit="" displayUnit="" format="" maxValue="" minValue="" maxAlarm="" minAlarm="" maxWarning="" minWarning="" deltaTime="" deltaValue=""/> + </attributes> + <attributes name="integration_time" attType="Scalar" rwType="READ" displayLevel="OPERATOR" polledPeriod="0" maxX="" maxY="" allocReadMember="true" isDynamic="false"> + <dataType xsi:type="pogoDsl:DoubleType"/> + <changeEvent fire="false" libCheckCriteria="false"/> + <archiveEvent fire="false" libCheckCriteria="false"/> + <dataReadyEvent fire="false" libCheckCriteria="true"/> + <status abstract="false" inherited="false" concrete="true" concreteHere="true"/> + <properties description="" label="" unit="" standardUnit="" displayUnit="" format="" maxValue="" minValue="" maxAlarm="" minAlarm="" maxWarning="" minWarning="" deltaTime="" deltaValue=""/> + </attributes> + <attributes name="time_stamp" attType="Scalar" rwType="READ" displayLevel="OPERATOR" polledPeriod="0" maxX="" maxY="" allocReadMember="true" isDynamic="false"> + <dataType xsi:type="pogoDsl:DoubleType"/> + <changeEvent fire="false" libCheckCriteria="false"/> + <archiveEvent fire="false" libCheckCriteria="false"/> + <dataReadyEvent fire="false" libCheckCriteria="true"/> + <status abstract="false" inherited="false" concrete="true" concreteHere="true"/> + <properties description="" label="" unit="" standardUnit="" displayUnit="" format="" maxValue="" minValue="" maxAlarm="" minAlarm="" maxWarning="" minWarning="" deltaTime="" deltaValue=""/> + </attributes> + <attributes name="pause_time" attType="Scalar" rwType="READ_WRITE" displayLevel="OPERATOR" polledPeriod="0" maxX="" maxY="" allocReadMember="true" isDynamic="false"> + <dataType xsi:type="pogoDsl:DoubleType"/> + <changeEvent fire="false" libCheckCriteria="false"/> + <archiveEvent fire="false" libCheckCriteria="false"/> + <dataReadyEvent fire="false" libCheckCriteria="true"/> + <status abstract="false" inherited="false" concrete="true" concreteHere="true"/> + <properties description="" label="" unit="" standardUnit="" displayUnit="" format="" maxValue="" minValue="" maxAlarm="" minAlarm="" maxWarning="" minWarning="" deltaTime="" deltaValue=""/> + </attributes> + <attributes name="visibilities" attType="Spectrum" rwType="READ" displayLevel="OPERATOR" polledPeriod="0" maxX="96" maxY="" allocReadMember="true" isDynamic="false"> + <dataType xsi:type="pogoDsl:DoubleType"/> + <changeEvent fire="false" libCheckCriteria="false"/> + <archiveEvent fire="false" libCheckCriteria="false"/> + <dataReadyEvent fire="false" libCheckCriteria="true"/> + <status abstract="false" inherited="false" concrete="true" concreteHere="true"/> + <properties description="" label="" unit="" standardUnit="" displayUnit="" format="" maxValue="" minValue="" maxAlarm="" minAlarm="" maxWarning="" minWarning="" deltaTime="" deltaValue=""/> + </attributes> + <attributes name="rcu_modes" attType="Spectrum" rwType="READ" displayLevel="OPERATOR" polledPeriod="0" maxX="96" maxY="" allocReadMember="true" isDynamic="false"> + <dataType xsi:type="pogoDsl:StringType"/> + <changeEvent fire="false" libCheckCriteria="false"/> + <archiveEvent fire="false" libCheckCriteria="false"/> + <dataReadyEvent fire="false" libCheckCriteria="true"/> + <status abstract="false" inherited="false" concrete="true" concreteHere="true"/> + <properties description="" label="" unit="" standardUnit="" displayUnit="" format="" maxValue="" minValue="" maxAlarm="" minAlarm="" maxWarning="" minWarning="" deltaTime="" deltaValue=""/> + </attributes> + <attributes name="raw_visibilities" attType="Image" rwType="READ" displayLevel="OPERATOR" polledPeriod="0" maxX="96" maxY="96" allocReadMember="true" isDynamic="false"> + <dataType xsi:type="pogoDsl:DoubleType"/> + <changeEvent fire="false" libCheckCriteria="false"/> + <archiveEvent fire="false" libCheckCriteria="false"/> + <dataReadyEvent fire="false" libCheckCriteria="true"/> + <status abstract="false" inherited="false" concrete="true" concreteHere="true"/> + <properties description="" label="" unit="" standardUnit="" displayUnit="" format="" maxValue="" minValue="" maxAlarm="" minAlarm="" maxWarning="" minWarning="" deltaTime="" deltaValue=""/> + </attributes> + <pipes name="pipe_visibilities" description="" label="" rwType="READ" displayLevel="OPERATOR"/> + <pipes name="pipe_raw_visibilities" description="" label="" rwType="READ" displayLevel="OPERATOR"/> + <preferences docHome="./doc_html" makefileHome="/usr/local/share/pogo/preferences"/> + </classes> +</pogoDsl:PogoSystem> diff --git a/StatsCrosslet-DS/requirements.txt b/StatsCrosslet-DS/requirements.txt new file mode 100644 index 0000000000000000000000000000000000000000..3b3b3b08bb560cfbe4fed2e7c7a0241f02f0af24 --- /dev/null +++ b/StatsCrosslet-DS/requirements.txt @@ -0,0 +1 @@ +opcua >= 0.98.9