FSM generator

Installation

bash cargo install -f fsm_gen

You can download from repository and build and install as usual

bash cargo build --release cargo install

TODO

Aim

All computing processes consist of receiving an input and processing it by generating an output. From the simplest to the most complex.

Sometimes, the process depends on the context and it is necessary to manage a state.

Messages and state management, therefore, are two fundamental elements of any software process.

That's why I wrote two external DSLs long time ago for both elements (which I still use in production and are a great help).

In this repository I rewrite one of them, the code generator for a state machine.

At the moment it generates C++ code (my most immediate target in production).

You can have data (fields) on inputs structs, and also on each status

To explain the system, I will use the example in cpp_test/fsm

This example is about writing a system that will handle login requests.

First the server will be asked for a password.

This key will be used to encode the username and password (this one will be passed through a hash function) in the login request.

This encoding will be irreversible (hash function). The server will perform the same operation (starting from the hash of the password) to verify the validity.

If it is OK, it will send a login confirmation.

The diagram would look like this:

Basic diagram

A list of transitions could be written as:

```peg [init] rqkey -> wlogin / sendkey timer -> init _ -> logout / logerr

[wlogin] rqlogin & valid -> login / sendlogin rqlogin -> logout / logerr timer & timeoutwl -> logout / logerr timer -> wlogin _ -> logout / log_err

[login] rqlogout -> logout / sendlogout heartbeat -> login / updatehb timer & timeoutl -> logout timer -> login _ -> logout / log_err

[logout] timer -> logout _ -> logout / log_err ```

And this is the input for this tool to generate code

In fact, even the previous diagram has been generated from this DSL (it generated a graphviz dot file)

Elements

States

peg [init] ...

init, w_login, login ... are the states

Depending on the input and the state (with its values as will be seen later), the system will change to a new state.

Transition

peg rq_key -> w_login

If we receive an input (in this case rq_key) we go to next state (w_login)

INPUT

The elements received by the status machine.

peg INPUT v rq_key -> w_login

In the example they are rqkey, rqlogin, rq_logout, heartbeat and timer.

GUARDS

Functions that will be called depending on the status and input to decide the way forward.

peg GUARD v rq_login & valid -> login / send_login

In the example we have valid, timeout, ontime.

All transitions can be replicated with different guards, but always one of them has to be without guard (and the last one). Example:

peg rq_login & valid -> login / send_login rq_login -> logout / log_err

Final status

Behind the -> arrow is the state we will change to.

peg FINAL_STATUS v rq_key -> w_login

Actions

We can define an action to be performed when executing a transaction.

This will be after the final state and '/'.

peg ACTION v rq_login -> logout / log_err

In this example we have sendkey, sendlogin...

Special transition

In all states it is necessary to consider all inputs.

But it is very common that many transitions are the same (generally error cases).

This is marked with the input _

Consider the init status:

peg [init] rq_key -> w_login / send_key timer -> init _ -> logout / log_err

_ will be expanded to produce...

peg [init] rq_key -> w_login / send_key timer -> init rq_login -> logout / log_err heartbeat -> logout / log_err rq_logout -> logout / log_err

Therefore considering all possible inputs in this state

Usage

To get help...

bash fsm_gen --help

``` fsm_gen 0.1.1 jleahred jleahred@gmail.com

Generate code from a simple fsm file
To check the supported languages  --show_langs

USAGE: fsmgen [FLAGS] [OPTIONS] [fsmfiles]...

FLAGS: -d, --dot-graphviz Generate graphviz dot file -h, --help Prints help information --help-cpp Give me some information about generating cpp files -s, --show-langs Show supported languages generators -V, --version Prints version information

OPTIONS: -l, --lang Language to generate code (show available --show-langs) [default: cpp] -t, --threads Number of threads to use. 0 means all [default: 0]

ARGS: ... List of fsm files ```

The default language is c++ (and at the moment the only one)

You can run:

bash fsm_gen login.fsm

And it will generate the c++

You can pass a list of fsm files

bash fsm_gen login.fsm test/seller.fsm test/test2/lift.fsm

The code will be generated on same directory of original .fsm file

If your shell supports it you could run...

bash fsm_gen **/*.fsm

C++ code generation

Starting from the example login.fsm the system will create...

fsm_login_gen.h fsm_login_gen.cpp

You don't have to modify these files. You have to write your code on next files...

fsm_login.h fsm_login.cpp

These two files will be created if don't exist as a reference

fsmlogingen.h

Full code on cpptest/fsm/fsmlogin_gen.h

We are informed on when it was created.

This file is automatically generated and you don't need to modify it, but it's a good point to see the API and prepare the code you have to fill by hand

cpp // generated automatically 2019-03-22 11:24:40 // do not modify it manually

Headers and namespaces based on filename

```cpp

ifndef FSMFSMLOGINGENERATEDH

define FSMFSMLOGINGENERATEDH

include

include

namespace login { ```

Forward declaration of types and a tip to know where to fill it

First with status, and next the inputs

```cpp // ---------------------------------------------------- // M A N U A L

// declaration on fsmlogin.h // forwardstatusinfo struct initinfot; struct wlogininfot; struct logininfot; struct logoutinfot;

// in struct heartbeatt; struct rqkeyt; struct rqlogint; struct rqlogoutt; struct timert;

// M A N U A L // ---------------------------------------------------- ```

Internal clase forward declaration

cpp class BaseState; typedef std::shared_ptr<BaseState> SState;

The fsm class you have to instanciate

```cpp // ------------------- // F S M class Fsm { public: Fsm(); ~Fsm();

void in(const heartbeatt& in); void in(const rqkeyt& in); void in(const rqlogint& in); void in(const rqlogoutt& in); void in(const timert& in);

... } ```

Methods of Fsm class to be defined by hand

First the funtions to generate new state

Next the guards funtions

```cpp

// ---------------------------------------------------- // M A N U A L

// implementation in fsm_login.cpp

// status change functions logoutinfot in2logout(const heartbeatt& i); logoutinfot in2logout(const rqkeyt& i); wlogininfot in2wlogin(const rqkeyt& i); logininfot in2login(const rqlogint& i); logoutinfot in2logout(const rqlogint& i); logoutinfot in2logout(const rqlogoutt& i); initinfot in2init(const timert& i); logininfot in2login(const timert& i); logoutinfot in2logout(const timert& i); wlogininfot in2wlogin(const timer_t& i);

// guards to implement bool valid(const rqlogint& in, const wlogininfot& stinfo); bool timeout(const timert& in, const wlogininfot& stinfo); bool ontime(const timert& in, const logininfot& stinfo);

// M A N U A L // ----------------------------------------------------

```

fsmlogingen.cpp

Full code on cpptest/fsm/fsmlogin_gen.cpp

You must not modify this file, and it's not necessary to know much about it

fsm_login.h

Full code on cpptest/fsm/fsmlogin.h

In this file you have to declare the status info types, and input types

If the file doesn't exist, it will be created with empty data types

cpp // Code generated automatically to be filled manually // This file will not be updated by generator // It's created just the first time as a reference

The header guard based on fsm file and a reference to the generated header

```cpp

ifndef FSMLOGINH

define FSMLOGINH

include

include "fsmlogingen.h"

```

On namespace based on fsm file, we have the two types to declare

You can, of course, complete these types on cpp file if necessary

```cpp namespace login {

// status info types struct initinfot{}; struct wlogininfot{}; struct logininfot{}; struct logoutinfo_t{};

// input types struct heartbeatt {}; struct rqkeyt {}; struct rqlogint {}; struct rqlogoutt {}; struct timert {};

} // namespace login

endif // FSMLOGINH

```

fsm_login.cpp

Full code on cpptest/fsm/fsmlogin.cpp

This is the other file you have to maintain by hand.

This methods, are declared on .h and annotated you have to fill by hand.

```cpp // Code generated automatically to be filled manually // This file will not be updated by generator // It's created just the first time as a reference

include "fsm_login.h"

include "fsmlogingen.h"

namespace login {

//  status change functions
logout_info_t in2logout(const heartbeat_t& i){}
logout_info_t in2logout(const rq_key_t& i){}
w_login_info_t in2w_login(const rq_key_t& i){}
login_info_t in2login(const rq_login_t& i){}
logout_info_t in2logout(const rq_login_t& i){}
logout_info_t in2logout(const rq_logout_t& i){}
init_info_t in2init(const timer_t& i){}
login_info_t in2login(const timer_t& i){}
logout_info_t in2logout(const timer_t& i){}
w_login_info_t in2w_login(const timer_t& i){}


//  guards
bool valid(const rq_login_t& in, const  w_login_info_t& st_info) { return true; }
bool timeout(const timer_t& in, const  w_login_info_t& st_info) { return true; }
bool on_time(const timer_t& in, const  login_info_t& st_info) { return true; }

} // namespace login ```

login

You will have a template that will be called on every state transition.

```cpp // log template void log(const std::string &txttrans, const IN &, const INITST &, const ENDST &) { std::cout << txttrans << std::endl; }

} // namespace login

endif // FSMLOGINH

```

You can specialize it as much as you want.

The first parameter is a text with the description of the transition. Example:

cfg [init] rq_key -> w_login [w_login] rq_login(valid) -> login [login] rq_logout -> logout

Diagrams source

...