bash
cargo install -f fsm_gen
You can download from repository and build and install as usual
bash
cargo build --release
cargo install
Update idata in order to use ispush with btrees and more
Complete the cpp example and update on README.md
output with signals
Add languages
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:
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)
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.
peg
rq_key -> w_login
If we receive an input (in this case rq_key
) we go to next state (w_login
)
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.
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
Behind the -> arrow is the state we will change to.
peg
FINAL_STATUS
v
rq_key -> w_login
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...
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
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
ARGS:
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
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
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
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 // ----------------------------------------------------
```
Full code on cpptest/fsm/fsmlogin_gen.cpp
You must not modify this file, and it's not necessary to know much about it
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
```
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
```
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
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 ```
You will have a template that will be called on every state transition.
```cpp
// log
template
} // namespace login
```
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
...