Skip to content

Commit

Permalink
Browse files Browse the repository at this point in the history
  • Loading branch information
bandophahita committed Oct 16, 2023
1 parent bb4ea34 commit c22b49d
Show file tree
Hide file tree
Showing 2 changed files with 152 additions and 3 deletions.
7 changes: 4 additions & 3 deletions screenpy/pacing.py
Original file line number Diff line number Diff line change
Expand Up @@ -19,11 +19,12 @@
def function_should_log_none(func: Function) -> bool:
"""Helper function to decide when to log return values.
Determine if function is attached to one of the protocols that allow for anything
to return.
Determine if function wrapped in beat should log that it returns None
"""
if func.__annotations__ and "return" in func.__annotations__:
return func.__annotations__["return"] is not None
anno = func.__annotations__["return"]
return anno is not None and anno != "None"

return False


Expand Down
148 changes: 148 additions & 0 deletions tests/test_pacing_future.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,148 @@
"""This is a duplicate of test_pacing but uses future annotations"""
from __future__ import annotations
import logging
from typing import Optional

from screenpy import Actor, IsEqualTo, See, act, aside, beat, scene


def prop():
"""The candlestick in the hall!"""


class Prop:
"""The wrench in the study!"""

def __init__(self, weapon: str, room: str, perpetrator: str) -> None:
self.weapon1 = weapon
self.room = room
self.perpetrator = perpetrator

@beat("The {weapon1} in the {room}!")
def use(self):
pass


class NonesyQuestion:
@beat("{} examines NonesyQuestion")
def answered_by(self, _: Actor) -> object:
return None

def describe(self) -> str:
return "NonesyQuestion"


class CornerCase:
@beat("{} examines CornerCase")
def answered_by(self, _: Actor) -> object:
self.do_not_log_me_man()
self.does_return_something()
self.no_annotations_rt_none(False)
self.no_annotations_rt_int(True)
return None

@beat("Blah!")
def do_not_log_me_man(self) -> None:
return None

@beat("Foobar...")
def does_return_something(self, toggle: bool = False) -> Optional[int]:
if toggle:
return 1
return None

# purposfully not annotated
@beat("Baz?")
def no_annotations_rt_none(self, toggle=False):
if toggle:
return 1
return None

# purposfully not annotated
@beat("Bazinga!!")
def no_annotations_rt_int(self, toggle=False):
if toggle:
return 1
return None

def describe(self) -> str:
return "CornerCase"


class TestAct:
def test_calls_narrators_method(self, mocked_narrator) -> None:
test_act = "Test Act"
actprop = act(test_act)(prop)

actprop()

mocked_narrator.announcing_the_act.assert_called_once()
announced_act = mocked_narrator.announcing_the_act.call_args_list[0][0][1]
assert test_act == announced_act


class TestScene:
def test_calls_narrators_method(self, mocked_narrator) -> None:
test_scene = "Test Scene"
sceneprop = scene(test_scene)(prop)

sceneprop()

mocked_narrator.setting_the_scene.assert_called_once()
set_scene = mocked_narrator.setting_the_scene.call_args_list[0][0][1]
assert test_scene == set_scene


class TestBeat:
def test_interpolations(self, mocked_narrator) -> None:
"""(This also tests that the narrator's method was called.)"""
test_weapon = "rope"
test_room = "ballroom"
test_prop = Prop(test_weapon, test_room, "")

test_prop.use()

mocked_narrator.stating_a_beat.assert_called_once()
completed_line = mocked_narrator.stating_a_beat.call_args_list[0][0][1]
assert completed_line == f"The {test_weapon} in the {test_room}!"

def test_beat_logging_none(self, Tester, caplog):
caplog.set_level(logging.INFO)
See(NonesyQuestion(), IsEqualTo(None)).perform_as(Tester)

assert [r.msg for r in caplog.records] == [
"Tester sees if nonesyQuestion is equal to <None>.",
" Tester examines NonesyQuestion",
" => <None>",
" ... hoping it's equal to <None>.",
" => <None>",
]

def test_beat_logging_none_corner(self, Tester, caplog):
caplog.set_level(logging.INFO)
See(CornerCase(), IsEqualTo(None)).perform_as(Tester)

assert [r.msg for r in caplog.records] == [
"Tester sees if cornerCase is equal to <None>.",
" Tester examines CornerCase",
" Blah!",
" Foobar...",
" => <None>",
" Baz?",
" Bazinga!!",
" => <1>",
" => <None>",
" ... hoping it's equal to <None>.",
" => <None>",
]


class TestAside:
def test_calls_narrators_method(self, mocked_narrator) -> None:
test_whisper = "<whisper whisper>"

aside(test_whisper)

mocked_narrator.whispering_an_aside.assert_called_once()
completed_line = mocked_narrator.whispering_an_aside.call_args_list[0][0][0]
assert completed_line == test_whisper

0 comments on commit c22b49d

Please sign in to comment.