Skip to content

I don't understand how to compose Intents/Effects together properly #84

@ckp95

Description

@ckp95

I'm trying to get my head around this library and I'm having some trouble understanding how I'm meant to structure Effects that are built out of other Effects, and how to test them. Let's say I have an Intent that represents getting text input from the user:

@dataclass
class GetInput:
    value: str = None

I write a @do-annotated generator function for it:

@do
def get_input(prompt):
    the_input = yield Effect(GetInput(prompt))
    return the_input

And I test it like this:

def test_get_input():
    prompt = "enter something: "
    seq = [
        (GetInput(prompt), lambda x: "The Quick Brown Fox")
    ]
    
    effect = get_input(prompt)
    assert perform_sequence(seq, effect) == "The Quick Brown Fox"

So far so good. I also want to have an Effect that represents printing something to the screen, so I write this:

@dataclass
class PrintString:
    value: str = None
    

@do
def print_string(value):
    yield Effect(PrintString(value))
    

def test_print_string():
    to_print = "something"
    seq = [
        (PrintString("something"), noop)
    ]
    
    effect = print_string(to_print)
    assert perform_sequence(seq, effect) is None

That works fine too.

If I want to combine these two, with some additional processing (converting to lowercase) I know I can do something like this:

@do
def print_lowercased_input(prompt):
    the_input = yield get_input(prompt)
    lowered = the_input.lower()
    yield print_string(lowered)


def test_print_lowercased_input():
    prompt = "enter something: "
    seq = [
        (GetInput(prompt), lambda x: "The Quick Brown Fox"),
        (PrintString("the quick brown fox"), noop)
    ]
    
    effect = print_lowercased_input(prompt)
    assert perform_sequence(seq, effect) is None

But how do I encapsulate these two Effects into one "thing", in a way that can be tested without having to know about the individual sub-Effects that comprise it? In other words, what I would like is to have this:

@dataclass
class PrintLowercasedUserInput:
    prompt: str = None

@do
def combined_effect(prompt):
    yield Effect(PrintLowercasedUserInput(prompt))
    

def test_combined_effect():
    prompt = "enter something: "
    seq = [
        (PrintLowercasedUserInput(prompt), lambda x: "the quick brown fox"),
    ]
    
    effect = combined_effect(prompt)
    assert perform_sequence(seq, effect) is None

In such a way that PrintLowercasedUserInput somehow encapsulates the action of the print_lowercased_input generator.

I'm finding it really hard to express what I want with the right vocabulary because this is all quite new to me. Basically when I'm writing tests for a large application I don't want the seq in the test to be full of very low level actions, I want to group them together into logical units, each of which has tests itself. Am I making any sense?

Metadata

Metadata

Assignees

No one assigned

    Labels

    No labels
    No labels

    Type

    No type

    Projects

    No projects

    Milestone

    No milestone

    Relationships

    None yet

    Development

    No branches or pull requests

    Issue actions