<div dir="ltr">Jamie, thanks for taking the time to detail out your proposal.<div><br></div><div>I have no major disagreements. The proposal looks fine to me.</div></div><div class="gmail_extra"><br><br><div class="gmail_quote">
On Thu, May 22, 2014 at 6:21 AM, Jamie Hannaford <span dir="ltr"><<a href="mailto:jamie.hannaford@rackspace.com" target="_blank">jamie.hannaford@rackspace.com</a>></span> wrote:<br><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">







<div style="word-wrap:break-word">
<p>
</p><div>Hey everyone,</div>
<div><br>
</div>
<div>Based on our conversation in irc yesterday, I’m detailing a few proposals for the way we handle testing. Before that, I want to establish the terminology so we’re all on the same page. In a software project like this, there are generally
 three types of tests: unit tests, integration tests, end-to-end tests.</div>
<div><br>
</div>
<div><b><br>
</b></div>
<div><b>End-to-end testing</b> is the full run-through of a service operation as an end-user would interact with it. You pass in your identity parameters, instantiate a service like Swift, and execute an operation. A real HTTP request
 is sent over the wire, and a real HTTP response from the server is received. In other words, it’s a live network test of
<i>all components </i>from end to end. Right now, any time we’re communicating with the server API in our test suite, it’s an
<i>end-to-end test</i>. There doesn’t need to be many of these - just enough to test that our API works for end-users. E2E tests will typically be slow (due to network calls) - but this does not matter.</div>
<div><br>
</div>
<div><b><br>
</b></div>
<div><b>Integration testing </b>is like end-to-end testing except no network connections happen. All it does is test is the
<i>integration </i>between modules of the application. So if we want to test Swift operation - we’d instantiate a context object with identity parameters, then instantiate a Swift service object, and then after we’ve done the setup, finally test the operation
 object. In an integration test, the flow of execution happens like it would an end-to-end test, but all we’re testing is that different components work together. This is useful for ensuring that contracts between interfaces are being satisfied, etc. I don’t
 think we need to worry about writing these.</div>
<div><br>
</div>
<div><br>
</div>
<div><b>Unit testing</b> is very different from both of the above. Instead, you test extremely small “units” of behavior in a particular class. Each test needs to be fully isolated and have only 1 responsibility (i.e. test one unit). The
 class you’re testing should not collaborate with real objects; instead, you need to pass in mocks. So, if we’re unit testing a Swift operation, instead of using a real service or transport client - we mock them and use the mock objects in our tests. If our
 tested class invokes methods on this mock, we also need to explicitly define how it does so. For example, if we’re testing a method that calls `$this->client->foo()` internally and expects a response, we need to explicitly tell the mocked client object to
 return a value when its “foo” method is called. We can also be more granular and strict: we can say that the method should only be called with certain arguments, that the method should be called x number of times. With unit tests, you are defining and testing
 communication promises. The point of doing this is that you’re testing HOW your object communicates in a precise way. Because they’re isolated, there should be hundreds of unit tests, and they should be very quick.</div>

<div><br>
</div>
<div><br>
</div>
<div>Here’s an example of how you’d mock for unit tests in phpunit: <a href="https://gist.github.com/jamiehannaford/ad7f389466ac5dcafe7a" target="_blank">https://gist.github.com/jamiehannaford/ad7f389466ac5dcafe7a</a></div>

<div><br>
</div>
<div>There was a proposal made that we should just inject a HTTP client that doesn’t make real transactions - but that is not feasible because it goes against what a unit test is. If you were to do this, your tested class would be executing
 UNMOCKED methods against a REAL object - making as many calls as it wants. It would not be isolated, and for that reason it would not be a unit test. It would be an
<i>integration test</i> because you’re forcing your tested class to interact like it would in the wild. Instead, you should mock every collaborator and explicitly define which calls are made against them by your tested class.</div>

<div><br>
</div>
<div>There are amazing libraries out there like Prophecy and phpspec which mocking a whole load easier and more natural - but I assume nobody wants to move away from phpunit…</div>
<div><br>
</div>
<div><b><br>
</b></div>
<div><b>Proposal going forward</b></div>
<div><br>
</div>
<div>So, here are my proposals for our current library:</div>
<div><br>
</div>
<div>1. Refactor our unit tests to use mocking instead of real HTTP/network calls. If a class relies on dependencies or collaborators, they will need to be mocked.</div>
<div><br>
</div>
<div>2. As services are added (Swift, Nova, Keystone), end-to-end tests are added for each. We’d therefore ensure that our SDK is interacting with the real API as expected.</div>
<div><br>
</div>
<div>3. Never use the @depends annotation on unit tests, because it makes them tightly coupled with each other and brittle. A unit test is supposed to be completely autonomous and independent - it should never depend on the output of another
 test. </div>
<div><br>
</div>
<div>4. Use the “setUp” and “tearDown” helper methods to easily set up test fixtures</div>
<div><br>
</div>
<div>5. In our source code, we need to make use of Dependency Injection AS MUCH AS POSSIBLE because it’s easier to test. If we don’t (choosing to directly instantiate objects in our code), it introduces tight coupling and is extremely
 hard to mock.</div>
<div><br>
</div>
<div><br>
</div>
<div><br>
</div>
<div>Does anybody have any major disagreements with my above proposal?</div>
<div><br>
</div>
<div><br>
</div>
<div>Jamie</div>
<p></p>
<p> </p>
<p>
</p><table border="0" cellpadding="0" width="504">
  <tbody>
  <tr>
    <td style="WIDTH:270px"><span style="font-family:Verdana;font-size:small">Jamie Hannaford</span><br><span style="font-family:Verdana;font-size:x-small">Software Developer III - CH</span></td>
    <td style="WIDTH:281px"><img alt="experience Fanatical Support" align="right" width="159" height="17"></td></tr>
  <tr>
    <td colspan="2"><img alt="LINE" width="504" height="4"></td></tr>
  <tr>
    <td><span style="font-family:Calibri"><table><tbody><tr><td><span style="font-family:Verdana;font-size:x-small">Tel: </span></td><td><span style="font-family:Verdana;font-size:x-small"><a href="tel:%2B41434303908" value="+41434303908" target="_blank">+41434303908</a></span></td>
</tr><tr><td><span style="font-family:Verdana;font-size:x-small">Mob: </span></td><td><span style="font-family:Verdana;font-size:x-small"><a href="tel:%2B41791009767" value="+41791009767" target="_blank">+41791009767</a></span></td>
</tr></tbody></table></span></td>
    <td><img alt="Rackspace" width="280" height="60"></td></tr>
  <tr>
    <td colspan="2"><img width="504" height="3"></td></tr></tbody></table><p></p>
<p> </p>
<p></p><span style="font-size:11px">Rackspace International GmbH a company registered in the Canton of Zurich, Switzerland (company identification number CH-020.4.047.077-1) whose registered office is at Pfingstweidstrasse 60, 8005 Zurich, Switzerland. Rackspace International GmbH privacy policy can be viewed at <a href="http://www.rackspace.co.uk/legal/swiss-privacy-policy" target="_blank">www.rackspace.co.uk/legal/swiss-privacy-policy</a><br>
-<br>Rackspace Hosting Australia PTY LTD a company registered in the state of Victoria, Australia (company registered number ACN 153 275 524) whose registered office is at Suite 3, Level 7, 210 George Street, Sydney, NSW 2000, Australia. Rackspace Hosting Australia PTY LTD privacy policy can be viewed at <a href="http://www.rackspace.com.au/company/legal-privacy-statement.php" target="_blank">www.rackspace.com.au/company/legal-privacy-statement.php</a><br>
-<span style="font-size:11px"></span><br>Rackspace US, Inc, 5000 Walzem Road, San Antonio, Texas 78218, United States of America</span><br><span style="font-size:11px">Rackspace US, Inc privacy policy can be viewed at <a href="http://www.rackspace.com/information/legal/privacystatement" target="_blank">www.rackspace.com/information/legal/privacystatement</a></span><br>
<span style="font-size:11px">-</span><br><span style="font-size:11px">Rackspace Limited is a company registered in England & Wales (company registered number 03897010) whose registered office is at 5 Millington Road, Hyde Park Hayes, Middlesex UB3 4AZ.</span><br>
<span style="font-size:11px">Rackspace Limited privacy policy can be viewed at <a href="http://www.rackspace.co.uk/legal/privacy-policy" target="_blank">www.rackspace.co.uk/legal/privacy-policy</a></span><br><span style="font-size:11px">-</span><br>
<span style="font-size:11px">Rackspace Benelux B.V. is a company registered in the Netherlands (company KvK nummer 34276327) whose registered office is at Teleportboulevard 110, 1043 EJ Amsterdam.</span><br><span style="font-size:11px">Rackspace Benelux B.V privacy policy can be viewed at <a href="http://www.rackspace.nl/juridisch/privacy-policy" target="_blank">www.rackspace.nl/juridisch/privacy-policy</a></span><br>
<span style="font-size:11px">-</span><br><span style="font-size:11px">Rackspace Asia Limited is a company registered in Hong Kong (Company no: 1211294) whose registered office is at 9/F, Cambridge House, Taikoo Place, 979 King's Road, Quarry Bay, Hong Kong.</span><br>
<span style="font-size:11px">Rackspace Asia Limited privacy policy can be viewed at <a href="http://www.rackspace.com.hk/company/legal-privacy-statement.php" target="_blank">www.rackspace.com.hk/company/legal-privacy-statement.php</a></span><br>
<span style="font-size:11px">-</span><br><span style="font-size:11px">This e-mail message (including any attachments or embedded documents) is intended for the exclusive and confidential use of the individual or entity to which this message is addressed, and unless otherwise expressly indicated, is confidential and privileged information of Rackspace. Any dissemination, distribution or copying of the enclosed material is prohibited. If you receive this transmission in error, please notify us immediately by e-mail at <a href="mailto:abuse@rackspace.com" target="_blank">abuse@rackspace.com</a> and delete the original message. Your cooperation is appreciated.</span></div>



</blockquote></div><br></div>