Source code for event_model.documents.event_descriptor

from typing import Any, Dict, List, Optional, Tuple, Union

from typing_extensions import Annotated, Literal, NotRequired, TypedDict

from .generate.type_wrapper import Field, add_extra_schema

Dtype = Literal["string", "number", "array", "boolean", "integer"]


class RdsRange(TypedDict):
    """RDS (Read different than set) parameters range.


    https://tango-controls.readthedocs.io/en/latest/development/device-api/attribute-alarms.html#the-read-different-than-set-rds-alarm
    """

    time_difference: Annotated[
        float,
        Field(
            description=(
                "ms since last update to fail after if set point and "
                "read point are not within `value_difference` of each other."
            )
        ),
    ]
    value_difference: NotRequired[
        Annotated[
            float,
            Field(
                description=(
                    "Allowed difference in value between set point and read point "
                    "after `time_difference`."
                )
            ),
        ]
    ]


class LimitsRange(TypedDict):
    low: Optional[float]
    high: Optional[float]


class Limits(TypedDict):
    """
    Epics and tango limits:
    see 3.4.1 https://epics.anl.gov/base/R3-14/12-docs/AppDevGuide/node4.html
    and
    https://tango-controls.readthedocs.io/en/latest/development/device-api/attribute-alarms.html
    """

    control: NotRequired[Annotated[LimitsRange, Field(description="Control limits.")]]
    display: NotRequired[Annotated[LimitsRange, Field(description="Display limits.")]]
    warning: NotRequired[Annotated[LimitsRange, Field(description="Warning limits.")]]
    alarm: NotRequired[Annotated[LimitsRange, Field(description="Alarm limits.")]]
    hysteresis: NotRequired[Annotated[float, Field(description="Hysteresis.")]]
    rds: NotRequired[Annotated[RdsRange, Field(description="RDS parameters.")]]


_ConstrainedDtype = Annotated[
    str,
    Field(
        description="A numpy dtype e.g `<U9`, `<f16`",
        pattern="[|<>][tbiufcmMOSUV][0-9]+",
    ),
]

_ConstrainedDtypeNpStructure = List[Tuple[str, _ConstrainedDtype]]


[docs] class DataKey(TypedDict): """Describes the objects in the data property of Event documents""" limits: NotRequired[Annotated[Limits, Field(description="Epics limits.")]] choices: NotRequired[ Annotated[List[str], Field(description="Choices of enum value.")] ] dims: NotRequired[ Annotated[ List[str], Field( description="The names for dimensions of the data. Null or empty list " "if scalar data", ), ] ] dtype: Annotated[ Dtype, Field( description=( "The type of the data in the event, given as a broad " "JSON schema type." ) ), ] dtype_numpy: NotRequired[ Annotated[ Union[_ConstrainedDtype, _ConstrainedDtypeNpStructure], Field( description=( "The type of the data in the event, given as a " "numpy dtype string (or, for structured dtypes, array)." ) ), ] ] external: NotRequired[ Annotated[ str, Field( description="Where the data is stored if it is stored external " "to the events", pattern=r"^[A-Z]+:?", ), ] ] object_name: NotRequired[ Annotated[ str, Field(description="The name of the object this key was pulled from."), ] ] precision: NotRequired[ Annotated[ Optional[int], Field( description="Number of digits after decimal place if " "a floating point number" ), ] ] shape: Annotated[ List[int], Field(description="The shape of the data. Empty list indicates scalar data."), ] source: Annotated[ str, Field(description="The source (ex piece of hardware) of the data.") ] units: NotRequired[ Annotated[Optional[str], Field(description="Engineering units of the value")] ]
class PerObjectHint(TypedDict): """The 'interesting' data keys for this device.""" fields: NotRequired[ Annotated[ List[str], Field(description="The 'interesting' data keys for this device."), ] ] NX_class: NotRequired[ Annotated[ str, Field( description="The NeXus class definition for this device.", pattern=r"^NX[A-Za-z_]+$", ), ] ] class Configuration(TypedDict): data: NotRequired[ Annotated[Dict[str, Any], Field(description="The actual measurement data")] ] data_keys: NotRequired[ Annotated[ Dict[str, DataKey], Field( description="This describes the data stored alongside it in this " "configuration object." ), ] ] timestamps: NotRequired[ Annotated[ Dict[str, Any], Field(description="The timestamps of the individual measurement data"), ] ] EVENT_DESCRIPTOR_EXTRA_SCHEMA = { "patternProperties": {"^([^./]+)$": {"$ref": "#/$defs/DataType"}}, "$defs": { "DataType": { "title": "DataType", "patternProperties": {"^([^./]+)$": {"$ref": "#/$defs/DataType"}}, "additionalProperties": False, }, }, "additionalProperties": False, } @add_extra_schema(EVENT_DESCRIPTOR_EXTRA_SCHEMA) class EventDescriptor(TypedDict): """Document to describe the data captured in the associated event documents""" configuration: NotRequired[ Annotated[ Dict[str, Configuration], Field( description="Readings of configurational fields necessary for " "interpreting data in the Events.", ), ] ] data_keys: Annotated[ Dict[str, DataKey], Field( description="This describes the data in the Event Documents.", title="data_keys", ), ] hints: NotRequired[PerObjectHint] name: NotRequired[ Annotated[ str, Field( description="A human-friendly name for this data stream, such as " "'primary' or 'baseline'.", ), ] ] object_keys: NotRequired[ Annotated[ Dict[str, Any], Field( description="Maps a Device/Signal name to the names of the entries " "it produces in data_keys.", ), ] ] run_start: Annotated[ str, Field(description="Globally unique ID of this run's 'start' document.") ] time: Annotated[ float, Field(description="Creation time of the document as unix epoch time.") ] uid: Annotated[ str, Field(description="Globally unique ID for this event descriptor.", title="uid"), ]