Skip to content

1. Creating a Schematic

Ovan Crone edited this page Dec 19, 2017 · 17 revisions

What is a schematic?

A schematic is the definition of a process (state-flow). You will define a schematic once for a certain purpose, such as how to send an email, make an API call, or more complicated flows. That schematic can then be used as the source code for instances of that flow called REstateMachines, or simply "Machines".

How to make a schematic

REstate provides a SchematicBuilder that helps guide you through creating a schematic from scratch. Let's first take a look at what we will be building, then we will go line by line, breaking it down.

The final code will be:

var schematic = REstateHost.Agent
    .CreateSchematic<string, string>("LoggerMachine")
    .WithState("Created", state => state
        .AsInitialState())
    .WithState("Ready", state => state
        .WithTransitionFrom("Created", "log")
        .WithReentrance("log")
        .WithOnEntry("log info", onEntry => onEntry
            .DescribedAs("Logs the transition as a message.")
            .WithSetting(
                key: "messageFormat", 
                value: "{schematicName}({machineId}) entered {state} on {input}. " +
                       "Message: {payload}")))
    .Build();

Creating a schematic

To begin we call the create schematic method.

var schematic = REstateHost.Agent
    .CreateSchematic<string, string>("LoggerMachine")

The type parameters on the method correlate to state and input types to be used in the schematic, as seen in the definition: CreateSchematic<TState, TInput>. The only parameter is the name of the schematic.

Defining an initial state

Next we need to define our initial state, the state the machine will be in when it is created.

    .WithState("Created", state => state
        .AsInitialState())

The state value is the string "Created". State values should be unique.

The second parameter is a lambda that allows you to modify the definition of the state generated. Here we use the .AsInitialState() method.

Next we can define another state, but this time with an action: a transition from the "Created" state, and a special type of transition called a re-entrance, which is to say there is a transition from the state into itself, like a loop or recursion.

Defining a state

First, we define the state "Ready" and start the state modifier lambda like before.

    .WithState("Ready", state => state

Transitions

Then, we can define a transition from our first state '"Created"' into our new state "Ready" when the machine receives the input "log".

        .WithTransitionFrom("Created", "log")

Re-Entrant transitions

Then, we can allow the re-entrant transition, "Ready" -> "Ready" when the machine receives the input "log".

        .WithReentrance("log")

Now that we have the acceptable transitions defined, we can add an action that should execute when the machine transitions into the state.

The first parameter is known as the ConnectorKey. It is just a string that corresponds to a connector (and its associated configuration) that has been registered with REstate. REstate ships with a LogEntryConnector that has several configurations available: "log info", "log debug", "log trace", "log warn", "log error", "log critical", where the second word is the logging level to use. For this example we use the "log info" connector key since our log message provides some "business value".

The second parameter is a lambda that lets us further define the OnEntry action, just like we did previously on state.

        .WithOnEntry("log info", onEntry => onEntry

OnEntry action

An OnEntry action can be given a description to make it less cryptic.

            .DescribedAs("Logs the payload as a message.")

OnEntry Settings

Going further, we can add some settings to our connector. The settings the connector will use vary per connector, so refer to their documentation for what is available and how to use them, but we can take a look at the log connector now.

The log connector has just one possible setting: messageFormat. When provided it lets you specify the format of the message that will be logged. There are a number of allowed special tokens such as {schematicName} that will be replaced at run-time with the current machine's values.

            .WithSetting(
                key: "messageFormat", 
                value: "{schematicName}({machineId}) entered {state} on {input}.")))

Building the Schematic

Finally to wrap it up, we can call the .Build() method to convert the SchematicBuilder<TState, TInput> to the serializer-friendly Schematic<TState, TInput>.

    .Build();

What's next?

We will continue with the schematic we just built and Create a StateEngine to store it.