Automata is a lightweight framework that allows users to define a state machine based websocket server.
Project description
Automata
Table of Contents
Introduction
Automata is a lightweight framework that allows users to define a state machine based websocket server. Each machine has a collection of states, for example, a traffic light state machine has Green, Yellow, and Red states. Each state can have any number of events, these events can be defined by the developer to do whatever they want. They can return data back to the client, transition to a new state, process data, or even do nothing; that is all up the the desired implementation. Overall, the goal of Automata is to provide a unified methodology and doctrine for defining and improving the clarity of websocket communication.
Getting Started
This is a simple example of how you to set up your Automata instance and run your application.
Prerequisites
python version >= 3.7
Installation and Setup
-
Install with
pip install automata-ws
-
Initialize your Automata (we use app.py)
from automata import Automata
my_machine = Automata(
name="myMachine"
)
- Create your states and any events (in any directory you want)
from automata import State, EventStatus, transition, transmit
red_light = State('red_light')
@red_light.event('change_to_green')
async def handler(automata, data):
#do something here...
run_some_func()
await transition(automata, 'green', EventStatus.OK) #will change the state of our machine and will report this to the client
green_state = State(
name="green",
targets=["yellow"]
)
@green_light.event('change_to_yellow')
async def handler(automata, data):
await transmit(automata, EventStatus.OK, some_data) #will send back data without changing state
#and so on and so on...
- Register states in your machine (back in the file you defined the machine)
my_machine.register_state(red_light) #the first state that is registered will be the initial state for new sessions
my_machine.register_state(green_light)
- Run the machine
my_machine.run('localhost', 8000)
States, Events, State Changes, Data Events, and Payloads
1. States
-
Every state must be unique: there cannot exist two states with the same name
-
A state may have a list of target states: all of the states in which a transition from the referenced state is possible
- Trying to transition to a state that is not in the target list for the referenced state will raise an exception
- An undefined target list means that a transition to any state is allowed
green_state = State( name="green", targets=["yellow"] #the target list with all the states that the green_state can transition to )
-
The initial state is the one that is registered first in the
Automata
instance
2. Events
- Events defined for a specific state must be unique
- All events are identified by their name and defined by their handler
- Each handler will recieve an
Automata
instance and adata
instance (either as adict
or astr
)
@green_state.event('change_to_yellow') #this annotation will create an event for the corresponding state (green_state in this case)
async def handler(automata: Automata, data):
run_some_code()
do_whatever_you_want()
await transition(automata, 'yellow', EventStatus.OK)
@green_state.event('ping')
def handler(automata: Automata, data): #the handler doesn't necessarily have to be async
print('pong')
3. State Changes
- Each state transition has to be relayed to the client: they will receive data about their new state and all events they can call
- Each state transition has to be accompanied with a status code: these are similar to the 100, 200, 300, 400, 500 status codes defined in HTTP and are found in the
EventStatus
enum - A state transition may have data associated with it if needed (data is optional)
- A state transition is achieved through the
transition
function-
await transition(automata, 'logged_in', EventStatus.OK, some_data) #a state transition with some data
-
4. Data Events
- It is possible to send data back to the client without changing state: this is achieved through the
transmit
function - Each data event must be accompanied with a status code (as mentioned above)
-
await transmit(automata, EventStatus.OK, some_data) #a data event with no state transition
-
5. Payloads
- All client to server payloads must follow the format below
-
{ "event" : "event_name", "data" : data }
-
- A state change will be relayed to the client in the following format (occurs when
transition
is called)-
{ "state" : "new_state", "events" : [ "event1", "event2", //Any number of possible events associated with the new state ], "status" : 200, "data": "some_data" //optional: can be None/Null }
-
- Any generic data event will be returned back in the following format (occurs when
transmit
is called)-
{ "status" : 200, "data": "some_data" //optional }
-
Advanced
1. Custom HTTP Endpoint(s)
- As of now all websocket connections are served on
/
- It is possible to implement a custom endpoint using the
automata.endpoint('/endpoint')
annotation - This is useful if a health check (or something generic) is needed
-
@my_machine.endpoint('/health') async def some_handler(request_headers): return 'OK'
-
- Note: only
get
requests can be handled
License
Apache License 2.0
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
File details
Details for the file automata-ws-0.1.1.tar.gz
.
File metadata
- Download URL: automata-ws-0.1.1.tar.gz
- Upload date:
- Size: 10.9 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/4.0.2 CPython/3.11.0
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | 85614a00f1a482d0a7f54e17b26fc70caf935c4e17f40bd12c5d3d812ee4fc7c |
|
MD5 | 5d11837dd8e970ee2f074587245d60c8 |
|
BLAKE2b-256 | ce6b40ab37e14ecfc82e32595d7ec14be0ab22f833498e14a719be89a49df330 |
File details
Details for the file automata_ws-0.1.1-py3-none-any.whl
.
File metadata
- Download URL: automata_ws-0.1.1-py3-none-any.whl
- Upload date:
- Size: 8.2 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/4.0.2 CPython/3.11.0
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | 6580eb5dbf17182d79be1ea687a93c660a31493869bfece51fbc7106e67077e0 |
|
MD5 | 192a3c0b20b200e6b50645c5adb0e7f9 |
|
BLAKE2b-256 | ca70cc833ec09695c1c3ce7f0591e9deac62756b935c9715302c19e8d8d91261 |