Friday, June 29, 2007 - 09:12

Practical Testdriven Development

We never learned much about testing our code when I was at university - the tutors told us about trying out boundary cases and that sort of thing, and for some tutorials we maybe had to draw up a simple test plan or describe how we tested the program; and the testing phase was mentioned in the Software Lifecycle module, but that was about it. No mention of automated testing, unit testing, etc. Partly that was just the way things were done - this was just before Java was starting to be widely accepted, and we were coding in c/c++.

So I always used to test my code manually, trying to run through everything I could think of. Then a couple of months into my 2nd job we decided to start doing unit testing with nUnit, and I hated it. Partly because I'd already written a bunch of code and now had to go back and write unit tests for it, which is always boring. Partly because a lot of the stuff I was doing wasn't really well suited for automated unit testing, being GUI stuff and stuff that was dependent on initial settings on a piece of hardware which I didn't have control over. And partly because because I hadn't designed my code in a way that allowed for easy unit testing. Now, though, I use it as an integral part of my dev process. It's still not quite testdriven, since the idea there is that you design and write your tests, with expected inputs and outputs, before writing your code; I find that difficult to do, because I think best about what I want the code to do while I'm writing, not sitting with a pencil and a piece of paper.

But what I do is write a chunk of code, then go write unit tests. I find that that's when I start thinking about things like, what if this is null? Or, what should actually happen in this case? I use the unit tests not just for automated regression testing, but also an initial way to exercise my code - it's actually than creating a temporary test gui to run each of your methods, which is what I mostly used to do. And then you have the advantage of being able to easily rerun your tests to make sure new code or changes hasn't broken anything.

Another benefit is that your unit tests can serve as an example of what what the code is supposed to do, what kinf of input it's intended to take, and the kind of values it returns. Sure, this should be described in your xml method comments, but sometimes an example just explains it so much more clearly. And if someone needs to know what your code will do in some odd situation, they don't have to wade through the source or hope that your documentation covers that case, they can just look at the unit tests (and hope that they cover that case).

In theory, unit testing is whitebox testing - you should exercise every line, every branch of your code. In practice, I try to do as much as I can but don't worry too much about about it all, since that can be really tricky, and can take more time than it's really worth.

I often don't write any asserts until after I've run the tests, which isn't quite the way it should work, I know. But it's easier to run the test, see what output you get, figure out if it's what you want (and if not, what you do want), and then write the asserts to match. Not exactly the testdriven philosophy, but I find it works well for me.

One of my initial arguments with unit testing was that you shouldn't write unit tests for your own code - if you think of writing a test for something, you've most likely thought of that issue while coding and have dealt with it. It's the things that you haven't thought of that you need to test, and someone else is more likely to see those things than you are. I still think this is true to an extent, but I tend to go into 'test writing mode' where I do think about things that wouldn't have occurred to me while coding - somehow I seem to switch mindsets.

I mentioned above that originally my code wasn't well designed for unit testing, and this is a point that I often find difficult. I don't believe that you should write and design your code with the ease of unit testing in mind; but generally, following good design principles does lead to easily tested code (short, simple methods;
loosely coupled classes, etc). The biggest issue, for me, is private methods. I refuse to make private methods public just so that I can unit test them; and unfortunately you don't have friends in c#. In a way it doesn't matter, since all your private methods must be used some public method at some point, otherwise it needn't exist, so by testing all your public methods you will test your private methods as well. But it makes your unit bigger, and makes it more difficult to test every branch - specifically where you want to vary the inputs to the enclosed private method, in ways that wouldn't come up in normal execution. And while you can use reflection to call your private methods, in practice it's just too clumsy and annoying to set up to do it foe every single method. I don't have a solution for this one yet.

Update: There's a good discussion about testing private methods here, although there is a shorter way to use reflection to invoke private methods, described here - basically, get the type of the class you want to test, call GetMethod() on the type to get the relevant MethodInfo, then call Invoke on the MethodInfo. Simple.

Update (29/06/2007): I came across a very good discussion on 'designing for tests' vs 'testing what's designed', and whether you should make methods public just so that they can be tested.

I've also started working with RhinoMocks, a mocking framework. This helps to keep your tests modular, and helps with some of the issues I've mentioned above. If you're testing methods in Class A, and they call something in Class B, you just mock out Class B telling it what you expect in return when you pass in a specific value. Sometimes this seems like you end up duplicating tests: you could just write a test that calls Class A's method, which calls Class B, and check you get the right value, and avoid writing tests on both Class A and Class B. But that's merging units, and you should test the two issues separately - testing Class A's reaction to what Class B returns, and testing what Class B generates based on the input from Class A, are two different things.

Of course, you're actually testing Class A's reaction to what you think Class B will return - if Class B was written by someone else, it may not return values you would expect (you may think it returns an empty string rather than null, for example). And that's where using Class B instead of a mock of Class B would reveal issues. On the other hand, is that what a unit test should be testing? Or does it rather belong in a higher level integration test?

RhinoMocks also lets you perform interaction based testing, as opposed to just state based testing. In other words, instead of checking what's returned, you can also check that certain methods, in various mocked classes, were or weren't called during the test. This is obviously pretty powerful, but can also get really complex. Especially when you're coming from a state based testing mindset.

I haven't quite got the hang of mocking frameworks yet - they seem like a really cool idea, but in practice, for non-trivial examples, I find it's often quite tricky. Sometimes it seems easier to just set up a higher level system test using something like Fitnesse. And sometimes that's okay, but sometimes you really need the unit level tests as well.

Labels:

Wednesday, June 27, 2007 - 13:56

Tiscali

I'm very pissed off with Tiscali - I've been trying to get a refund from them since I cancelled my account more than 3 months ago. It's only a small amount, but dammit, it's my money and not theirs, and I want it back! (At the very least, I don't want them to have it - I'd rather they send it to a charity than just give up and let them have it). Here's a summary of what happened - I'll try to keep it short, but there was a lot of bouncing backwards and forwards so it'll be a long post.

At least in this exchange I mostly manage to avoid the script-people. But because each time you reply it's picked up by a different person, you lost a lot of continuity. Also, there's no incentive for them to solve the problem - they know that when you reply it'll be someone else's issue, so why put any effort into trying to sort it out?

Here goes. In February, I phoned Tiscali to cancel my account as of 15 March, told them I was moving overseas, and gave them my non-UK forwarding address. At no time was this a problem, nor did they mention any potential limitations on refunds.

At the end of March they took a full month's fee from my bank account, and I emailed them to query this, since it should only have been a pro-rata amount. I struggled to find the billing email address, and got stuck in a loop with the script people who said I have to phone the billing department. Eventually I got the email address, but by then I had found info on the web which suggested that they take a full month then give you a refund at the end of the next month.

By 11 May, I hadn't received a refund so I emailed them. Person One replied saying that I am due a refund, and they would post me a cheque. I immediately replied asking them to rather refund directly to my bank account since it's very expensive to deposit foreign currency cheques in SA. Person Two responded saying they can only refund by cheque, or I can email them my credit card details and they would refund to that. Admittedly I misread his reply slightly (mentally added in a 'not' where there shouldn't have been one) and replied asking them to refund to my bank account. Person Three replied saying that I could phone them to have the refund sent to my credit card, since they couldn't accept credit card details via email (contradicting Person Two). I replied explaining why I can't phone and can't accept a cheque (since the forex fees would be too high), and asked if there is no way it could be refunded to my bank account. Person Four then replied asking me to phone them so they can refund back to my credit card. I replied, again saying that I cannot phone since international calls are extremely expensive, and asked if the cheque could be sent to someone else, made out to some else, or sent directly to my bank.

On 12 May, Person Five replied asking me to write to their head office, and saying that the cheque could only be made out to me. I also received a reply from Person Six, saying that they can make the cheque out to anyone I request. She suggested that I write to their head office to request the cheque to be sent directly to my bank. I replied saying that if I have to post something to the UK regardless, they might as well send the cheque to me here, and I could decide how to proceed from there. I again gave them my non-UK address.

On 16 May, I emailed them request a confirmation that the cheque was being processed. By 21 May, I had received no reply so I emailed again, again giving them my non-UK address.

On 22 May, Person Seven said she would put through an escalation to the billing department, but it could take 28 days to be processed.

By 25 June, I still had not received a cheque. I emailed them again, asking them to look into whether the cheque had been processed, and what address it had been sent to. On 26 June, Person Eight replied saying that the cheque was sent to my UK address on 23 May, and that a note had been attached to my account saying that it should be sent to SA instead. She said that if I cannot get hold of the cheque sent to the UK, they would cancel it and send a replacement to my SA address. I replied asking them to do so, giving them my SA address again (and complaining about how ridiculous the whole situation was). I received a reply from Person Six, asking if I have received the cheque since my last email, and if not, to advise them of the address I want the cheque sent to. I replied with a copy of my previous reply.

I then received a reply from Person Nine, saying that they cannot send cheques overseas, and I must confirm if I have a family member in the UK who they can send a cheque to who can then forward it on to me. At this point I nearly freaked out, and replied saying that this was not acceptable, that I do not have any family members in the UK, that no other company had had a problem refunding either to my bank account or sending a cheque to SA, and I did not not understand why it could not be done, especially since no-one had previously said it would it be a problem. I asked her to sort it out ASAP or escalate the issue to someone who could sort it out.

I have had no reply today. I don't think I can take another reply from them - we're just going backwards. I've going to email this summary to them, asking them to escalate to the Customer Service Manager or General Manager as per their Code of Practice, but I just know it'll end up having no effect - I'll just keep floating around the support pool. I'd take them up on their offer of mailing the cheque direct to my bank, but since I have to do it in writing they'll probably lose or ignore my letter, or send it to the wrong bank, or not include my account details, or wait another month and then tell me they can't do it.

Edit (28/06/2007): I've removed the names from the story, since one of the Tiscali customer service reps was uncomfortable with it (see the 3rd comment below).

I've since asked them just to send the cheque direct to my bank - Person Six replied saying that they will do so, and that they apologise for the problems and that they take on board all comments with a view to improving their services. I wonder if that includes laughing at their customers? (See the 1st comment below).

It's become two - no, three - issues: getting my money; getting contradictory answers; attitude towards customers. But
I just want to get the money I'm owed. The customer service reps clearly don't care, and I can't force them to care. The head of customer service should care, and the general manager should care, but I've been fighting with them for over three months and I'm too tired.

Labels:

Tuesday, June 26, 2007 - 09:52

Stay Away From Acapulco Spur

I have had a really crappy day. So I'm in rant mode, not just about things that happened today (not all which I can blog about), but going back to Saturday. So there'll be more ranting posts over the next few days - I apologise in advance, but sometimes you can't help but vent. I'll follow up with a fun cartoon when I'm done, to make up for it!

For this post: Acapulco Spur in Bergvliet. Apparently they are busy renovating, but are too scared of losing business to close the Spur.

We were shown to a table which was sitting on a plain concrete floor, and had to walk past building materials to get to it. There were no lights in this section, and we had only one candle for a fairly large table. We had to take turns using the candle to try to read the menus! The table and seats were filthy with building dust, and when we asked for the tables and seats to be cleaned the waiter brought two wet wipes to wipe the table (with which he wiped half the table, only). We then started wondering how clean the mats, cutlery, salads, etc were!

The menus themselves are impossible to read in good light, never mind in the light of a single candle. We have complained about them repeatedly in previous months, and nothing has been done. The printing has rubbed off to the point where the menus cannot be read - the prices are 'blob blob. blob blob', which is simply unacceptable. On some menus, if you know what you want, you can find the item and attempt to read the price; but if you want to browse the menu to see what's available, well, you're out of luck because large sections are simply blank.

At this point we got up and left. I didn't want to spoil what could have been a nice evening out by eating there, and I didn't feel I could condone their staying open in such a mess. And I certainly didn't want to pay good money for a shoddy experience. And with only one candle, I doubt we would have been able to see what we were eating anyway.

I emailed Spur about this, via their website. They passed the query on to the manager, who basically said that they've been operating for 35 years; they're very forward thinking; and they have a notice up aplogising for inconvience during the renovations. In other words, he basically said "tough, we don't care". Which makes me even less inclined to go back there. As I replied to him - inconvenience is sections being closed off, or having a longer waiting time. Dirt is not an inconvenience, it's a health hazard. The only good thing he had to say is that their new menus are being delivered on Thursday - sorry, too little, too late, and too little customer care.

Update: I've since been contacted by someone higher up, and they apologised profusely for the service and the reply. I've also spoken to the manager, who apologised for the renovations and explained how they're having a hard time but are almost done, and he's offered to make reparations. So while I think they were wrong, it happened and they're doing their best to make it right, which I appreciate.

Labels:

Monday, June 18, 2007 - 08:22

Cape Town vs London

It's taken a while, but as promised, here are the photos to show why I prefer working in Cape Town to working in London :-) These were all taken from around my office in London:

London London
London

And Cape Town:

Waterfront Waterfront
Waterfront Waterfront


I'm not going to get into the relative advantages and merits of each place - I'd have a list that goes on forever! I do miss London now that I'm in Cape Town; but I missed Cape Town when I was in London. The difference is that I miss London less from Cape Town than I missed Cape Town from London, if you know what I mean. What it comes down to is that Cape Town is home, and while London is great, it's not home for me. Although there are two things I particularly miss at the moment - central heating, and Green & Black's Hot Chocolate! If it didn't come in a great big heavy glass bottle, I'd consider requesting some via post for my birthday next month ;)

And while I'm posting photos, here are some totally gratuitous photos of my car (a Kia Picanto, which I'm very happy with):

Kia Picanto Kia Picanto

Labels: ,