This is Part 8 of a series on using State Machines to express complex logic.

One of our previous posts looked at how to have some logic fire when entering a state.

In this post, we will look at how to have some logic fire when exiting a state.

Suppose our Call system supported playing music to users who are on hold.

Given we are designing flexible, maintainable software, we want to be able to swap put streaming services:

We thus define our contract, via an interface:

public interface IStreamingService
{
    // Get the current volumn
    byte Volume { get; }
    // Increase the volume by one step
    void IncreaseVolume();
    // Decease the volume by one step
    void DecreaseVolume();
    // Mute the music
    void Mute();
    // Unmute the music
    void Unmute();
}

We then implement one for Spotify.

public class SpotifyStreamer : IStreamingService
{
    public byte Volume { get; private set; }

    public void IncreaseVolume()
    {
        if (Volume < 10)
            Volume++;
    }

    public void DecreaseVolume()
    {
        if (Volume > 0)
            Volume--;
    }

    public void Mute()
    {
        Volume = 0;
    }

    public void Unmute()
    {
        Volume = 5;
    }
}

We then inject this into our Call class so that when placed on Hold, the music is activated, and when leaving Hold, it is muted.

public sealed class Call
{
    private readonly IStreamingService _service;

    public int MusicVolume => _service.Volume;

    // State machine
    private readonly StateMachine<Status, Trigger> _stateMachine;

    public Call(IStreamingService service) : this(Status.Ready, service)
    {
    }

    private Call(Status status, IStreamingService service)
    {
        _service = service;
        _stateMachine = new StateMachine<Status, Trigger>(status);
        service.Mute();

        //
        // Configure state machine
        //
        _stateMachine.Configure(Status.Ready)
            .Permit(Trigger.Dial, Status.Ringing);

        _stateMachine.Configure(Status.Ringing)
            .Permit(Trigger.PickUp, Status.Connected)
            .Permit(Trigger.HangUp, Status.Ready)
            .OnEntry(() => Log.Information("Ringing..."));

        // Only allow transition of the time since start time is greater than or
        // equal to the threshold (10 seconds). This is by capturing the current time
        // at the point of requested state change and comparing with the start time

        _stateMachine.Configure(Status.Connected)
            .Permit(Trigger.Hold, Status.OnHold)
            .Permit(Trigger.HangUp, Status.Ready)
            .OnEntry(() => { Log.Information("Connected..."); });


        _stateMachine.Configure(Status.OnHold)
            .PermitIf(Trigger.UnHold, Status.Connected)
            .PermitIf(Trigger.HangUp, Status.Ready)
            .OnEntry(() =>
                {
                    Log.Information("Placing On Hold...");
                    Log.Information("Un-muting streamer currently at volume {Volume}...", service.Volume);
                    service.Unmute();
                }
            ).OnExit(() =>
            {
                Log.Information("Exiting hold...");
                Log.Information("Muting streamer currently at volume {Volume}... ...", service.Volume);
                service.Mute();
            });

        _stateMachine.Configure(Status.Ready)
            .OnEntry(() => Log.Information("Hanging Up..."));
    }

    public void Dial()
    {
        _stateMachine.Fire(Trigger.Dial);
    }

    public void HangUp()
    {
        _stateMachine.Fire(Trigger.HangUp);
    }

    public void PickUp()
    {
        _stateMachine.Fire(Trigger.PickUp);
    }

    public void Hold()
    {
        _stateMachine.Fire(Trigger.Hold);
    }

    public void UnHold()
    {
        _stateMachine.Fire(Trigger.UnHold);
    }
}

Finally, we write some tests to verify everything functions correctly.

[Fact]
public void Music_Is_Played_On_Hold_And_Muted_After()
{
    var spotify = new SpotifyStreamer();
    var call = new Call(spotify);
    call.Dial();
    call.PickUp();
    call.MusicVolume.Should().Be(0);
    call.Hold();
    call.MusicVolume.Should().Be(5);
    call.UnHold();
    call.MusicVolume.Should().Be(0);
}

If we run our tests …

ExitStateTestResults

The console prints the messages we expect:

ExitStateLogs

In this way, we can have code that takes advantage of state changes before or after.

TLDR

Stateless offers an OnExit() method that you can use to write logic to fire just before the transition exits.

The code is in my GitHub.

Happy Hacking!