It's my first job as a web developer so I'm still a bit "lost".
Today I tried to write unit test for this class in order to learn how to use Phake test doubles framework:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<?php | |
class SendingFormByEmailAndRedirecting | |
{ | |
private $form; | |
private $email; | |
private $formRedirection; | |
function __construct( | |
FormEmail $email, | |
Form $form, | |
FormRedirection $formRedirection | |
) { | |
$this->email = $email; | |
$this->form = $form; | |
$this->formRedirection = $formRedirection; | |
} | |
public function execute() | |
{ | |
if ($this->form->isNotFilled()) { | |
$this->formRedirection->backToFormPage(); | |
return; | |
} | |
$sendingSucceeded = $this->email->send( | |
$this->form->fields() | |
); | |
$this->redirect($sendingSucceeded); | |
} | |
private function redirect($sendingSucceeded) | |
{ | |
if ($sendingSucceeded) { | |
$this->formRedirection->toSuccessPage(); | |
} else { | |
$this->formRedirection->toFailPage(); | |
} | |
} | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<?php | |
class SendingFormByEmailAndRedirectingTest extends PHPUnit_Framework_TestCase | |
{ | |
private $form; | |
private $formRedirection; | |
private $email; | |
private $sendAndRedirectAction; | |
private $WHATEVER_FIELDS; | |
function setUp() | |
{ | |
$this->WHATEVER_FIELDS = | |
array("fieldName" => "fieldContent"); | |
$this->form = Phake::mock('Form'); | |
$this->formRedirection = Phake::mock('FormRedirection'); | |
$this->email = Phake::mock('FormEmail'); | |
$this->sendAndRedirectAction = | |
new SendingFormByEmailAndRedirecting( | |
$this->email, | |
$this->form, | |
$this->formRedirection | |
); | |
} | |
function testRedirectionBackToFormPage() | |
{ | |
$this->whenFormNotFilled(); | |
$this->sendAndRedirectAction->execute(); | |
Phake::verify($this->formRedirection) | |
->backToFormPage(); | |
} | |
function testEmailSending() | |
{ | |
$this->whenFormFilledWithWhateverFields(); | |
$this->sendAndRedirectAction->execute(); | |
Phake::verify($this->email) | |
->send($this->WHATEVER_FIELDS); | |
} | |
function testRedirectionToSuccesPage() | |
{ | |
$this->whenFormFilledWithWhateverFields(); | |
$this->andEmailSentSuccessfully(); | |
$this->sendAndRedirectAction->execute(); | |
Phake::verify($this->formRedirection)->toSuccessPage(); | |
} | |
function testRedirectionToFailPage() | |
{ | |
$this->whenFormFilledWithWhateverFields(); | |
$this->andEmailNotSentSuccessfully(); | |
$this->sendAndRedirectAction->execute(); | |
Phake::verify($this->formRedirection)->toFailPage(); | |
} | |
private function whenFormFilledWithWhateverFields() { | |
Phake::when($this->form) | |
->isNotFilled() | |
->thenReturn(false); | |
Phake::when($this->form) | |
->fields() | |
->thenReturn($this->WHATEVER_FIELDS); | |
} | |
private function whenFormNotFilled() | |
{ | |
Phake::when($this->form) | |
->isNotFilled() | |
->thenReturn(true); | |
} | |
private function andEmailSentSuccessfully() | |
{ | |
Phake::when($this->email) | |
->send($this->WHATEVER_FIELDS) | |
->thenReturn(true); | |
} | |
private function andEmailNotSentSuccessfully() | |
{ | |
Phake::when($this->email) | |
->send($this->WHATEVER_FIELDS) | |
->thenReturn(false); | |
} | |
} |
Although I like the readability of the tests, I'm not very happy with them because I think they will be too fragile. At the same time I'm not sure how else I could test the behavior of SendingFormByEmailAndRedirecting class...
To give you a bit more of context this is how I'm using it from my code:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<?php | |
$action = | |
Action::sendingByEmail() | |
->formIn("empresas") | |
->to("xxx@ppp.es") | |
->build(); | |
$action->execute(); |
I don't really like the name of the class but it's because I don't have any other context. It may be OK if you are using the Command pattern. As a consumer of the API I would expect the "form" object to be passed in to "execute".
ReplyDeleteMaybe the tests are trying to tell you something about your design. It could be an interesting experiment, using mocks rather than spies. In the tests, I would put lines 32 and 33 in different tests. Of just remove line 32. Ideally tests should fail for a single reason.
Interesting problem and interesting post, thank you :-)
Thanks for your feedback, Carlos.
DeleteI don't like the class name either, I'm still struggling with it.
I imagined the class as representing an "action" so the name tried to describe the action. I don't have more types of actions yet so I'm not using a Command pattern.
I'll try passing the form too see how it feels.
This will also allow me to remove the word form from the names of both the "action" and its collaborators.
You're totally right, I could perfectly remove line 32 (already removed).
I also added an example of how I'm using the class.
Thank you very much.
Best regards,
M