Skip to main content

API for interfacing with the core acquisition process via platform and language agnostic message queues.

Project description

Autotrainer API: Python Integration

The python auto-trainer-api module is intended to provide an efficient means to emit information that is needed for local or remote management of applications running locally on the device and to receive commands from those sources.

The exposed API is intended to be agnostic to the underlying transport layer. The current implementation uses ZeroMQ. The reasons for this decision include:

  • Relatively low overhead for the acquisition application
  • Does not require either side to manage connections and know when the other side is available or changes availability
  • Does not require an additional process or service to maintain a persistent message queue (e.g., RabbitMQ)
    • Persistent data is managed elsewhere

Client Integration

There are two points of integration available for clients to support the remote interface. The first allows publishing "events" for state and property changes that occur in the client. The second is a "command" interface for the client to receive command requests from remote sources.

Both interfaces are provided through an instance of the RpcService class. An instance can be obtained via create_api_service(...) which constructs the specific concrete implementation. After creation, the service must be explicitly started (start(...)) and can be stopped (stop(...)). Once stopped, an instance can not be restarted. If a connection should be reestablished after stopping an instance, a new instance should be created and started.

Events

Events are published through the send_event_dict(message) or send_event(message) methods on the RpcService instance.

send_event_dict accepts a dictionary with the following entries:

  • kind - an ApiEventKind value
  • when - a wall-clock value of time
  • index - monotonically increasing timestamp w/units of nanoseconds (typically time.perf_counter_ns())
  • context - an object whose contents depend on the ApiEventKind; may be None for some event kinds

send_event accepts an ApiEvent dataclass instance with the same fields.

Emergency events (ApiEventKind.emergencyStop, ApiEventKind.emergencyResume) are automatically published on both the standard event channel and a dedicated emergency channel for subscribers that only monitor critical events.

Commands

Commands from external sources are supported by registering a CommandRequestDelegate with the RpcService instance via the command_request_delegate property. The delegate receives an instance of ApiCommandRequest and must return an instance of ApiCommandRequestResponse.

The primary property of the ApiCommandRequest is command which is an ApiCommand value. Depending on the command, there may also be a dictionary in the data property with arguments or other information relevant to the command. The nonce property can be ignored if the command is handled synchronously. For commands that send an asynchronous result after completion (see below), the nonce must be stored to associate with the result (along with the command value).

The returned ApiCommandRequestResponse object contains one required field:

  • result - a value of ApiCommandRequestResult

And three optional fields:

  • data - an optional object with results from the command beyond success/failure (often will be None)
  • error_code - an integer error code value if the command is not successful or can't be initiated
  • error_message - an optional string message if the command is not successful or can't be initiated

The expected contents of the data property are defined by the command, but is typically None.

The error_code property should be a non-zero value if there is an error code to report.

There are two fields on the ApiCommandRequestResponse object that are ignored as part of the returned object from the command delegate: command and nonce. See Asynchronous Commands for when these fields are required.

Asynchronous Commands

The command delegate is expected to return "immediately" (low millisecond type of time frame). If the command is not deterministically fast, it is expected to immediately return an ApiCommandRequestResponse with a result value of ApiCommandRequestResult.PENDING_WITH_NOTIFICATION.

Once the action associated with the command is complete, the client should call send_command_result(response) on the RpcService instance. The response argument is an ApiCommandRequestResponse instance with the command and nonce properties set to the values received in the original ApiCommandRequest (the client is responsible for storing these values until needed). Note that those two properties are ignored for synchronous command handling, but required for asynchronous responses.

Bridge

The autotrainer.api.bridge module provides a Socket.IO bridge that allows web-based clients to interact with the RPC service without a direct ZeroMQ connection. It exposes a FastAPI/Socket.IO application that:

  • Forwards command requests from the RPC service to connected Socket.IO clients
  • Accepts command results from Socket.IO clients and relays them back to the RPC service
  • Allows Socket.IO clients to publish events via the sendApiEvent message

The bridge can be started as a standalone process:

python -m autotrainer.api.bridge --host 0.0.0.0 --port 8000

Or created programmatically via create_bridge_app(options, command_timeout, cors_allowed_origins) for embedding into a larger application. Requires the bridge optional dependency group (pip install auto-trainer-api[bridge]).

Tools

Client Application

scripts/client_application.py starts an interactive process that enables the RpcService as a "real" autotrainer application would. It generates heartbeat events, periodically publishes system status events, and responds to commands (GET_CONFIGURATION, GET_STATUS). Run with:

python scripts/client_application.py

The following interactive commands are available:

  • app_launch - publish an applicationLaunched event
  • e_stop - publish an emergencyStop event
  • e_resume - publish an emergencyResume event
  • detector / d <detector_id> <is_active> - update a detector's status and publish a detectorChanged event
  • alarm / a <alarm_id> <is_active> <is_enabled> - update an alarm's status and publish an alarmChanged event
  • magnet <intensity> - set the tunnel device magnet intensity (0.0-100.0)
  • baseline <intensity> - set the baseline magnet intensity (0.0-100.0) and publish a headfixBaselineChanged event
  • q / quit - stop the service and exit

Detector and alarm IDs can be specified by numeric value, exact enum name, or a unique prefix of the name (case-insensitive).

This is primarily useful for testing remote applications/services without running the main autotrainer acquisition application.

Remote Console

scripts/remote_console.py starts an interactive process that connects to an RpcService instance in the same manner full remote management services would. Run with:

python scripts/remote_console.py [-H <host>]

It supports the -H/--host argument to specify a remote host address (defaults to 127.0.0.1). Available commands: start, stop, configuration, status, quit.

Publishing

Publishing is handled automatically via GitHub Actions. Pushing a version tag (e.g., v0.9.18) triggers the CI workflow which runs the test suite, builds the package, and publishes to PyPI using trusted publishing (OIDC).

Pushes to the workflows-edit branch publish to TestPyPI for verification.

Installation

The package is published to the PyPi package index and can be installed with standard pip commands.

pip install auto-trainer-api

Optional dependency groups:

pip install auto-trainer-api[bridge]     # Socket.IO bridge dependencies
pip install auto-trainer-api[telemetry]  # OpenTelemetry support

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

auto_trainer_api-0.9.20.tar.gz (29.5 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

auto_trainer_api-0.9.20-py3-none-any.whl (28.5 kB view details)

Uploaded Python 3

File details

Details for the file auto_trainer_api-0.9.20.tar.gz.

File metadata

  • Download URL: auto_trainer_api-0.9.20.tar.gz
  • Upload date:
  • Size: 29.5 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for auto_trainer_api-0.9.20.tar.gz
Algorithm Hash digest
SHA256 a59864a8a57bc2a5f03e90a705be9caada96c0d85f28d18d1d0a8a6226a0bae9
MD5 e3867a4759df712bf4ca20809afff04a
BLAKE2b-256 eba303d485aec40c983e97ff84cf2e56424843f1e560a2bf965db2ea448fd5b3

See more details on using hashes here.

Provenance

The following attestation bundles were made for auto_trainer_api-0.9.20.tar.gz:

Publisher: publish.yml on Mouse-GYM/auto-trainer-api

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file auto_trainer_api-0.9.20-py3-none-any.whl.

File metadata

File hashes

Hashes for auto_trainer_api-0.9.20-py3-none-any.whl
Algorithm Hash digest
SHA256 e05ec1cadfe95b7072c4e0d28adf14764f9533d92bb2248093f4df8ec8f3290b
MD5 a2842ed5bbfa7654d2507976a82d67d3
BLAKE2b-256 0b49864e5f04c883468f60074529639e80b69f327b9d24b8243ca160451c8b46

See more details on using hashes here.

Provenance

The following attestation bundles were made for auto_trainer_api-0.9.20-py3-none-any.whl:

Publisher: publish.yml on Mouse-GYM/auto-trainer-api

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page