Welcome to the code review for Takeaway Challenge! Again, don't worry - you are not expected to have all the answers. The following is a code-review scaffold for Takeaway Challenge that you can follow if you want to. These are common issues to look out for in this challenge - but you may decide to take your own route.
If you don't feel comfortable giving technical feedback at this stage, try going through this guide with your reviewee and review the code together.
Please checkout your reviewee's code and run their tests. Read the code and try some manual feature tests in IRB. How easy is it to understand the structure of their code? How readable is their code? Did you need to make any cognitive leaps to 'get it'?
Please do include all the gems you use in your Gemfile. This is an important courtesy to other developers and yourself in the future, so that all the project dependencies can be pulled in whenever the project is checked out on a new machine, e.g. twilio-ruby
gem
Every good code base will have its README updated following the contribution notes, i.e.
- Make sure you have written your own README that briefly explains your approach to solving the challenge.
- If your code isn't finished it's not ideal but acceptable as long as you explain in your README where you got to and how you would plan to finish the challenge.
The above is a relatively straightforward thing to do that doesn't involve much programming. Pro-tip: work on this while letting your sub-conscious work on those trickier coding problems :-)
The README is a great place to show the full story of how your app is used (from a user's perspective), e.g.
2.2.3 :001 > t = TakeAway.new
=> #<TakeAway:0x007f975b1a39a8 @menu=#<Menu:0x007f975b1a3890 @dishes={"spring roll"=>0.99, "char sui bun"=>3.99, "pork dumpling"=>2.99, "peking duck"=>7.99, "fu-king fried rice"=>5.99}>, @basket={}, @text_provider=#<TwilioAPI:0x007f975b1a33e0>>
2.2.3 :002 > t.read_menu
=> {"spring roll"=>0.99, "char sui bun"=>3.99, "pork dumpling"=>2.99, "peking duck"=>7.99, "fu-king fried rice"=>5.99}
2.2.3 :003 > t.order 'spring roll'
=> "1x spring roll(s) added to your basket."
2.2.3 :004 > t.order 'spring roll'
=> "1x spring roll(s) added to your basket."
2.2.3 :005 > t.order 'spring roll', 4
=> "4x spring roll(s) added to your basket."
2.2.3 :006 > t.basket_summary
=> "spring roll x4 = £3.96"
2.2.3 :007 > t.add 'pork dumpling', 3
=> "3x pork dumpling(s) added to your basket."
2.2.3 :008 > t.basket_summary
=> "spring roll x4 = £3.96, pork dumpling x3 = £8.97"
2.2.3 :009 > t.total
=> "Total: £12.93"
2.2.3 :010 > c.checkout(12.93)
You may have read about "Vacuous" tests in the airport challenge code review. The example there focused on how we shouldn't test the behaviour of a double; but we can get into similar trouble if we are stubbing a real object, e.g.
it 'sends a payment confirmation text message' do
expect(subject).to receive(:send_sms)
subject.send_sms(20.93)
end
In the above the expect(subject).to receive(:send_sms)
command "stubs" out any existing method called send_sms
on the subject. Using expect
instead of allow
means that at the end of the it block, RSpec checks that subject did receive the message send_sms
, which we have ensured by calling subject.send_sms
, so this test passes without ever touching the application code.
You can confirm this test is 'vacuous' by checking that the test coverage doesn't change when you remove it.
In general you shouldn't be stubbing out behaviour on the object under test. The two key exceptions are when you have randomness or a 3rd party API. We saw how to stub random behaviour in the airport challenge code review, but how do we stub a 3rd party API? See the next section.
The Twilio gem provides access to the online Twilio service. If we don't stub out this interaction, we will send test SMS messages every time we run our tests. Not a good thing.
The simplest approach is to stub out a method that calls the service, for example:
class Takeaway
def complete_order(price)
send_text("Thank you for your order: £#{total_price}")
end
def send_text(message)
# this method calls the Twilio API
end
...
end
can be stubbed out like so:
describe Takeaway
subject(:takeaway) { described_class.new }
before do
allow(takeaway).to receive(:send_text)
end
it 'sends a payment confirmation text message' do
expect(takeaway).to receive(:send_text).with("Thank you for your order: £20.93")
takeaway.complete_order(20.93)
end
end
This ensures that Takeaway#complete_order gets some test coverage and that no SMS will be sent by our tests. This is acceptable, but we still don't have very good test coverage. See the pill on levels of stubbing 3rd party services for some alternatives.
Note that if you create real objects (not doubles) in your unit tests other than that which is the subject, then you are using the Chicago style of unit testing (also called integration testing). In general you want to separate up your unit from your integration (or "feature") tests. Unit tests can just rest in the root of the spec folder, but features of integration tests should go in a subfolder (spec/features or spec/integration) or even in a separate folder on the root directory to allow them to run completely separately.
At Makers Academy we recommend using the London style with doubles to effectively isolate the single class being tested in a unit test.
# London Style
describe Order do
let(:menu) { double :menu, price: '£1.00', contains?: true }
subject(:order) { described_class.new(menu) }
it 'order total to be sum of items added' do
order.add('Pizza')
order.add('Pizza')
expect(order.total).to eq '£2.00'
end
end
# Chicago Style --> arguably an 'integration' or 'feature' test
describe Order do
subject(:order) { described_class.new(Menu.new) } # note real Menu class
it 'order total to be sum of items added' do
order.add('Pizza')
order.add('Pizza')
expect(order.total).to eq '£2.00'
end
end
The public interface of a class is any method that can be accessed publicly. So for example:
class Takeaway
def complete_order(price)
is_correct_amount?(price)
...
end
def is_correct_amount?(price)
total_price == price
end
end
has two public methods, complete_order
and is_correct_amount?
, that both should be tested independently is_correct_amount?
will be implicitly tested by any test of complete_order
, but since it is public it should have it's own expilcit test.
However, perhaps is_correct_amount?
will only ever used internally by Takeaway, and never called by any collaborator objects? In which case we can make the public interface of Takeaway simpler like so:
class Takeaway
def complete_order(price)
is_correct_amount?(price)
...
end
private
def is_correct_amount?(price)
total_price == price
end
end
private
methods do not require tests; although having too many private methods is a design smell. Anyhow, having moved is_correct_amount?
into a private method we have reduced the complexity of the public interface of Takeaway (a good thing) and we no longer have to test is_correct_amount?
explicitly, which also slims down our tests. Reducing the complexity of the public interface of any class (both in terms of number of methods, and numbers of arguments they take) is generally a good thing as it reduces the numbers of ways in which the object can interact with other objects, and encourages a loose coupling between objects that promotes re-use.
There are two main uses of modules in Ruby; one is to provide 'utility' libraries (which are sometimes a code smell) and the other is to provide mixins. However, using a module as a mixin can violate the Single Responsibility Principle. Although code is defined in the module, when it is include
d in a class, its behaviour becomes part of that class and therefore part of the class's responsibilities. Shared behaviour can be refactored into mixins (e.g. BikeContainer
in Boris Bikes), but other responsibilities the class is dependent on (e.g. sending text messages for the restaurant) should be injected (see this practical on dependency injection).
The Law of Demeter suggests that:
- Each object should have only limited knowledge about other units: only units "closely" related to the current unit.
- Each object should only talk to its friends; don't talk to strangers.
- Only talk to your immediate friends.
The following test shows a process of reaching through a series of related objects. The warning sign is the multiple periods in subject.menu.dishes.length
. Here we are seeing Restaurant
is being tested for properties that belong to the menu - effectively they are none of restaurant's business and shouldn't be tested here; and we shouldn't see deep-reaching chains like this in application code either:
describe Restaurant do
it "has 2 items on the menu" do
expect(subject.menu.dishes.length).to eq(2)
end
end
Note this is different to method chaining, which enables the calling of multiple methods on the same object in one line of code. The Demeter violation applies to reaching down through multiple objects.
It is likely that the Restaurant
(or equivalent) class is dependent on another object to handle the Twilio messaging. If not, then this is a violation of Single Responsibility Principle. In order to invert dependencies and make testing easier, the Twilio class should be injected into the Restaurant
class. So instead of:
class Restaurant
def initialize()
@mmessager = Messager.new
end
end
...
restaurant = Restaurant.new
You can have (note you can define a default for the dependency as shown here, but that's optional):
class Restaurant
def initialize(messager = Messager.new)
@messager = messager
end
end
...
restaurant = Restaurant.new
# or
restaurant = Restaurant.new(dummy_messager)
# where dummy_messager might be a test double for example
Applications generally comprise a number of concerns. For example, pure business logic is a concern; interacting with the user (UI) is a concern; persisting data to a file or database is a concern; and so on. Generally, as well as having a single responsibility, a class should only be involved in one concern (which kind of follows, right?).
To this end, a class that contains pure business logic should not also be concerned with the User Interface or presentation logic. If your business logic class uses puts
statements to communicate with the user, then it has poor separation of concerns. Business logic objects should return other objects and status indicators that can be translated in a separate presentation layer into user-friendly messages and interactions. This means our business logic is not constrained to a particular output representation.
Separation of concerns leads to some very powerful design patterns such as Model View Controller (MVC), which we will meet in Week 4.
# good
class Menu
def to_s
'the menu is empty'
end
end
$ menu = Menu.new
$ puts menu # note that the to_s is called by default when `puts`ing an object in Ruby
=> "the menu is empty"
# bad
class Menu
def display
puts 'the menu is empty'
end
end
$ menu = Menu.new
$ menu.display
=> "the menu is empty" # note that this string will be outputed in your tests etc.
It's easy to overlook responsibilities and end up with a class that does too much. This is a great opportunity to refactor your design to extract those responsibilities. One common indication is that a group of methods share a noun. For example, in Restaurant
we might have:
def add_to_order(item)
...
end
def order_total
...
end
def finalize_order
...
end
The noun 'order' appears in three method names and this is a clear indication that we need an Order
class. The beauty of OO is that as soon as we extract this responsibility into another class, our design becomes instantly much more powerful. Enabling the restaurant to handle multiple orders is suddenly much easier.
A well ordered codebase will use ENV variables and the dotenv gem to ensure that sensitive infomration such as phone numbers and security tokens are not pushed up to public repos on Github.
This can be particularly useful if you are managing counts of things (e.g. dishes). Instead of:
def initialize
@items = {}
end
def add_dish(dish, quantity = 1)
@items[dish] = 0 unless items[dish]
@items[dish] += quantity
end
You can remove the test and initialization in the first line of add_dish
by defining 0
as the default:
def initialize
@items = Hash.new(0)
end
def add_dish(dish, quantity = 1)
@items[dish] += quantity
end
instead of the following (which assumes @items
is an array rather than a hash):
def total_price
total = 0
@items.each do |item|
total += item.price
end
total
end
You can use the reduce
method (alias inject
) already provided by Ruby:
def total_price
@items.reduce { |sum, item| sum + item }
end
The Open Closed Principle tells us that we want our code to be open for extension but closed for modification. The idea is that if we need to add some new functionality then we can do that by extending our code rather than modifying it.
For example the menu items should not be hard coded into a restaurant class. Arguably they should not be in the business logic at all, e.g. being added at runtime (i.e. in IRB) or loaded from an external hash or maybe even a file. We are concerned that the menu items will likely change, so if they are hard coded in like this:
class Restaurant
def initialize
@menu = {}
@menu[:pizza] = 100
@menu[:coke] = 70
end
end
Then in order to make any changes to the menu we have to open up the Restaurant class itself. Restaurant is not open for extension at all in the above example. It must be opened up for modification in order to make any changes to the menu. Consider this alternative:
class Restaurant
def initialize(menu = Menu.new)
@menu = menu
end
end
The type of Menu that Restaurant uses can now easily be changed. It is "open for extension". We can create a new type of Menu class that loads from a file, takes dynamic user, whatever we want, and Restaurant will happily collaborate (assuming the menu has the correct public interface, e.g. Menu#contains?, Menu#price etc.). This latter Restaurant is also "closed for modification". We don't need to modify because it can easily be extended through the use of different collaborator objects.
The Ruby community has a very consistent style guide and you should follow it. Use tools like Rubocop (gem install rubocop ; rubocop
) to analyze your code for violations.
You may find it difficult to remember the indentation rules, but one helpful rule of thumb for indentation is to ensure that you are two space indented inside any do ... end
, class ... end
or def ... end
block.