The application that I have been assigned to create is a song library, an application that the product owner can use to keep track of her music collection. With this application she can add and remove songs, and for each song the application keeps track of the artist, the song title and the song length.
Wireframes
I am given some sketches showing what the final application might look like. As the product owner sees it, there will be four windows.
The main window displays all songs. It is called the Songs Window.
If the user clicks the Add button, a window opens where the user can specify the song title, artist and song length. It is called the Add Song Window. The user can choose to create the song by clicking OK or to discard it with cancel.
If the user clicks OK and creates the song, the song is presented in a new, read only window. It is called the View Song Window.
The same window can also be opened by double-clicking one of the rows in the song list in the Songs Window. If the user clicks the change button in this window a window opens where the user can change the song information. It looks the same as the Add Song Window, but it is called the Change Song Window.
When the user adds a new song, or modifies a song, all other windows should be updated with the new information.
Here is an image describing the flow of the application:
User Stories
Based on the user interface sketches, the product owner and I come up with the following user stories:
- Display song list, that is, the list of songs in the Songs Window
- Add new song
- Open song by double-clicking it in the list of songs
- Display song after adding it
- Change song information
- Persistence, that is, make sure that the song list is still there after restarting the application.
Other requirements
There are a few other features that are of a more functional kind that the application should have.
First of all, the product owner hopes to be able to sell the application to customers in other countries. That is why she would like to be able to translate the application into other languages, and also have the application conform to the locale used in that country.
There is also one thing that I bring up for the product owner that she hasn't thought about, but that she agrees we should consider. I would like to add a crash report feature. In the extremely unlikely event (famous last words) of a system crash, I, as a developer, would like to have as good information as possible about what went wrong. Included in this crash report I would like to add the stack trace and a log containing the user's last interactions with the system. The stack trace contains detailed information about what went wrong and the log file contains information about how to reproduce the crash. I call the logging Use Case Logging.
Terminology
Software development is, to quite a great extent, a communication activity. Instructing the computer what to do is a rather small part of the development, while the communication about the problem and the solution between product managers, end users, testers and developers, both verbally, in documentation and in code, is the major part.
To ease all this communication, I want to give concepts that concern the system a unique and unambiguous name. These names should be used in all verbal communication, but also in the code, when naming classes, methods, fields, etc. This is called our ubiquitous language. Agreeing on a vocabulary is one of the most powerful things you can do to simplify this communication.
Often, the language is something that already exists. It might be a general language, like in the case of the music business, or it might be a language only used within your organization. It might occur that the language does not exist, and you have to invent new words. This is something that you often do in product development.
In this case, all I have to do is use a language inspired by the music business. Here are terms that the domain expert and I agree on to use.
- Song library: The application.
- Song: A musical composition that the song library keeps track of
- Song title: The name of the song
- Artist: The name of the artist or band that performs the song
- Song length: The length of the song
- Add song: The action of adding a song to the song library.
- Display song: The action of opening and viewing a song from the song library.
The list of words found in this trivial example almost seems unnecessary to spend time on. But I'd argue that even with this small application this activity is important. For instance, imagine that, in my code, I sometimes use the word artist to name fields and parameters, sometimes band name and sometimes performer. What are the chances that a future reader of the code would be confused thinking that these might actully mean different things?
This list may or may not be complete, but most likely is not. As we come up with other important words I will extend our vocabulary.
I have also previously mentioned the windows that are part of the user interface. These also have their unique name that I will use consequently throughout the development.
Architecture, tools and frameworks
And here comes the section where I describe my big plans for a layered architecture, dividing the application into several services, using all of the latest and greatest frameworks, and so on.
But I won't. Instead I will let the design and architecture grow as I go along.
The same thing goes for testing strategy. I won't decide up front how I will write my tests. I will start writing the first test the way it feels right and continue adding tests where I find the need. Unit tests, integration tests, function tests, acceptance tests, whatever tests…
I will, however, have to make one (reversible) up-front decision, and that is what test framework to use. And I choose NUnit. I could have chosen some kind of specification tool, like specflow, which would allow including non-technical stakeholders in writing tests (or examples, as they are called when working with Specification by Example). This would be a really good idea, since it opens up for a richer communication. However, in this book I will write the tests and you, who are also a programmer, will read the tests. Both you and I are comfortable using a xUnit framework and working with C# code, and I choose the tool that is best suited for us. This book is however not about NUnit and most discussions can be applied also when working with other frameworks.
To enable a rapid feedback loop, I set up a build server that builds my code and runs all my tests when I check in the code. This is something that is not covered by this book, though, but with today's excellent build servers it takes me about an hour to set up.