// Copyright 2024 Christophe Henry // henry UNDERSCORE christophe AT hotmail DOT com // This is an extended version of the state machine available in the boost::mpl library // Distributed under the same license as the original. // Copyright for the original version: // Copyright 2005 David Abrahams and Aleksey Gurtovoy. Distributed // under the Boost Software License, Version 1.0. (See accompanying // file LICENSE_1_0.txt or copy at // http://www.boost.org/LICENSE_1_0.txt) // back-end #include //front-end #include #include #ifndef BOOST_MSM_NONSTANDALONE_TEST #define BOOST_TEST_MODULE back11_simple_with_puml_test #endif #include using namespace std; namespace msm = boost::msm; using namespace msm::front; using namespace msm::front::puml; namespace boost::msm::front::puml { // events template<> struct Event { int cpt_ = 0; }; template<> struct Event { int cpt_ = 0; }; template<> struct Event {}; template<> struct Event {}; template<> struct Event {}; // A "complicated" event type that carries some data. enum DiskTypeEnum { DISK_CD = 0, DISK_DVD = 1 }; template<> struct Event { Event(std::string name, DiskTypeEnum diskType) : name(name), disc_type(diskType) {} std::string name; DiskTypeEnum disc_type; }; // The list of FSM states template<> struct State : public msm::front::state<> { template void on_entry(Event const&,FSM& ) {++entry_counter;} template void on_exit(Event const&,FSM& ) {++exit_counter;} int entry_counter=0; int exit_counter=0; }; template<> struct State : public msm::front::state<> { template void on_entry(Event const&, FSM&) { ++entry_counter; } template void on_exit(Event const&, FSM&) { ++exit_counter; } int entry_counter = 0; int exit_counter = 0; }; template<> struct State : public msm::front::state<> { template void on_entry(Event const&, FSM&) { ++entry_counter; } template void on_exit(Event const&, FSM&) { ++exit_counter; } int entry_counter = 0; int exit_counter = 0; }; template<> struct State : public msm::front::state<> { template void on_entry(Event const&, FSM&) { ++entry_counter; } template void on_exit(Event const&, FSM&) { ++exit_counter; } int entry_counter = 0; int exit_counter = 0; }; template<> struct State : public msm::front::state<> { template void on_entry(Event const& e,FSM& ) { ++entry_counter; event_counter = e.cpt_; } template void on_exit(Event const&,FSM& ) {++exit_counter;} int entry_counter = 0; int exit_counter = 0; int event_counter=0; }; //actions template<> struct Action { template void operator()(EVT const&, FSM&, SourceState&, TargetState&) { } }; template<> struct Action { template void operator()(EVT const&, FSM&, SourceState&, TargetState&) { } }; template<> struct Action { template void operator()(EVT& e, FSM& fsm,SourceState& ,TargetState& ) { ++e.cpt_; ++fsm.test_fct_counter; } }; template<> struct Action { template void operator()(EVT const&, FSM& fsm, SourceState&, TargetState&) { ++fsm.start_playback_counter; } }; template<> struct Action { template void operator()(EVT const&, FSM& fsm, SourceState&, TargetState&) { fsm.process_event(Event{}); } }; template<> struct Action { template void operator()(EVT const&, FSM&, SourceState&, TargetState&) { } }; template<> struct Action { template void operator()(EVT const&, FSM&, SourceState&, TargetState&) { } }; template<> struct Action { template void operator()(EVT& e, FSM&, SourceState&, TargetState&) { ++e.cpt_; } }; template<> struct Action { template void operator()(EVT const&, FSM&, SourceState&, TargetState&) { } }; template<> struct Action { template void operator()(EVT const&, FSM&, SourceState&, TargetState&) { } }; // guard conditions template<> struct Guard { template bool operator()(EVT const&, FSM&, SourceState&, TargetState&) { return true; } }; template<> struct Guard { template bool operator()(EVT& evt, FSM&, SourceState&, TargetState&) { // to test a guard condition, let's say we understand only CDs, not DVD if (evt.disc_type != DISK_CD) { return false; } return true; } }; template<> struct Guard { template bool operator()(EVT const&, FSM&, SourceState&, TargetState&) { return true; } }; template<> struct Guard { template bool operator()(EVT const&, FSM& fsm, SourceState&, TargetState&) { ++fsm.can_close_drawer_counter; return true; } }; } namespace { // front-end: define the FSM structure struct player_ : public msm::front::state_machine_def { unsigned int start_playback_counter=0; unsigned int can_close_drawer_counter=0; unsigned int test_fct_counter=0; BOOST_MSM_PUML_DECLARE_TABLE( R"( @startuml Player skinparam linetype polyline state Player{ [*]-> Empty Stopped -> Playing : play / TestFct,start_playback [DummyGuard] Stopped -> Open : open_close / open_drawer Stopped -> Stopped : stop Open -> Empty : open_close / close_drawer [can_close_drawer] Empty --> Open : open_close / open_drawer Empty ---> Stopped : cd_detected / store_cd_info [good_disk_format && always_true] Playing --> Stopped : stop / stop_playback Playing -> Paused : pause / pause_playback Playing --> Open : open_close / stop_and_open Paused -> Playing : end_pause / resume_playback Paused --> Stopped : stop / stop_playback Paused --> Open : open_close / stop_and_open } @enduml )" ) // Replaces the default no-transition response. template void no_transition(Event const&, FSM&,int) { BOOST_FAIL("no_transition called!"); } }; // Pick a back-end typedef msm::back11::state_machine player; BOOST_AUTO_TEST_CASE( back11_simple_with_puml_test ) { player p; p.start(); BOOST_CHECK_MESSAGE(p.get_state&>().entry_counter == 1, "Empty entry not called correctly"); p.process_event(Event{}); BOOST_CHECK_MESSAGE(p.current_state()[0] == 1,"Open should be active"); //Open BOOST_CHECK_MESSAGE(p.get_state&>().exit_counter == 1,"Empty exit not called correctly"); BOOST_CHECK_MESSAGE(p.get_state&>().entry_counter == 1,"Open entry not called correctly"); p.process_event(Event{}); BOOST_CHECK_MESSAGE(p.current_state()[0] == 2,"Empty should be active"); //Empty BOOST_CHECK_MESSAGE(p.get_state&>().exit_counter == 1,"Open exit not called correctly"); BOOST_CHECK_MESSAGE(p.get_state&>().entry_counter == 2,"Empty entry not called correctly"); BOOST_CHECK_MESSAGE(p.can_close_drawer_counter == 1,"guard not called correctly"); p.process_event(Event{"louie, louie", DISK_DVD}); BOOST_CHECK_MESSAGE(p.current_state()[0] == 2,"Empty should be active"); //Empty BOOST_CHECK_MESSAGE(p.get_state&>().exit_counter == 1,"Open exit not called correctly"); BOOST_CHECK_MESSAGE(p.get_state&>().entry_counter == 2,"Empty entry not called correctly"); p.process_event(Event{"louie, louie", DISK_CD}); BOOST_CHECK_MESSAGE(p.current_state()[0] == 3,"Playing should be active"); //Playing BOOST_CHECK_MESSAGE(p.get_state&>().exit_counter == 2,"Empty exit not called correctly"); BOOST_CHECK_MESSAGE(p.get_state&>().entry_counter == 1,"Stopped entry not called correctly"); BOOST_CHECK_MESSAGE(p.get_state&>().exit_counter == 1,"Stopped exit not called correctly"); BOOST_CHECK_MESSAGE(p.get_state&>().entry_counter == 1,"Playing entry not called correctly"); BOOST_CHECK_MESSAGE(p.start_playback_counter == 1,"action not called correctly"); BOOST_CHECK_MESSAGE(p.test_fct_counter == 1,"action not called correctly"); p.process_event(Event{}); BOOST_CHECK_MESSAGE(p.current_state()[0] == 4,"Paused should be active"); //Paused BOOST_CHECK_MESSAGE(p.get_state&>().exit_counter == 1,"Playing exit not called correctly"); BOOST_CHECK_MESSAGE(p.get_state&>().entry_counter == 1,"Paused entry not called correctly"); // go back to Playing p.process_event(Event{}); BOOST_CHECK_MESSAGE(p.current_state()[0] == 3,"Playing should be active"); //Playing BOOST_CHECK_MESSAGE(p.get_state&>().exit_counter == 1,"Paused exit not called correctly"); BOOST_CHECK_MESSAGE(p.get_state&>().entry_counter == 2,"Playing entry not called correctly"); BOOST_CHECK_MESSAGE(p.get_state&>().event_counter == 1,"Playing event counter incorrect"); p.process_event(Event{}); BOOST_CHECK_MESSAGE(p.current_state()[0] == 4,"Paused should be active"); //Paused BOOST_CHECK_MESSAGE(p.get_state&>().exit_counter == 2,"Playing exit not called correctly"); BOOST_CHECK_MESSAGE(p.get_state&>().entry_counter == 2,"Paused entry not called correctly"); p.process_event(Event{}); BOOST_CHECK_MESSAGE(p.current_state()[0] == 0,"Stopped should be active"); //Stopped BOOST_CHECK_MESSAGE(p.get_state&>().exit_counter == 2,"Paused exit not called correctly"); BOOST_CHECK_MESSAGE(p.get_state&>().entry_counter == 2,"Stopped entry not called correctly"); p.process_event(Event{}); BOOST_CHECK_MESSAGE(p.current_state()[0] == 0,"Stopped should be active"); //Stopped BOOST_CHECK_MESSAGE(p.get_state&>().exit_counter == 2,"Stopped exit not called correctly"); BOOST_CHECK_MESSAGE(p.get_state&>().entry_counter == 3,"Stopped entry not called correctly"); } }