Tasting Tests at Cookpad

 

Keep on testing, all the way to your Realm apps

While you're in the testing groove, start testing your apps. Apps powered by Realm are easy to test, since the database is simple and is designed to be testable. Have confidence is every part of your code!

In this talk, Kazuaki Matsuo covers the testing methodology used at Cookpad, one of the most popular recipe sharing services in the world.


Introduction

I’m Kazuaki Matsuo. I work at Cookpad as a test engineer and software engineer in quality. My focus is on automation for mobile and improve the overall development process. I work with multiple languages: Swift, Ruby, Java for Android and Elixir.

Recently, I’ve been maintaining the Appium Ruby binding. If you’re unfamiliar, Appium is one of the most popular mobile test automation tools.

What Is Cookpad?

Cookpad is one of the most popular recipe sharing services in the world. Currently, we provide our services for Japan, and different countries around the world. According to similarweb.com, Cookpad is the largest site worldwide in the food category.

We have two iOS apps: one is for Japan and the other is for rest of the world.

Cookpad for iOS (Japan)

For the purpose of demonstration, the test target is the Japanese edition. The Japanese version of the app has been in development for a substantial period of time; it has around 100,000 lines of code. Our release cycle is around one month. Changes typically include many new features, and refactor to the internal logic.

The Kano model

The Kano model is a product development and customer satisfaction theory that is popular in Japan. The model has two main qualities: must-be quality, and attractive quality. Must-be quality is defined as passing basic standards and functionality, for example, in Japan, this means a crash-free app for almost all users, and no features can be degraded for a long time.

Mobile app environment has changed quickly through time. With each new version, user experiences require “quality by market”. I’ve come to think of this as diachronic quality. This word is influenced by linguistics. Diachronic means sometimes has developed and evolved through time.

Basic Strategy for Testing

To make the target testable, we should consider architecture for the apps from two aspects: internal and external. Internal is defined as the production cost side, and external is the UI/end user experience.

Check if the UI behavior is broken by rewriting and refactoring features, then implement and conduct unit tests. Developers can find bugs if their implementations break unexpectedly. UI testing includes manual checks. We’ll then be able to proceed re-engineering the app with this cycle step-by-step.

UI Tests Should Be Automated

One area that cries out for automation is UI testing. In many cases, UI tests take time and human resources because test automation for this area is difficult.

Converting manual checking into automated UI tests is important. If you’re able to increase your automated UI tests, you can automatically check most of the layers, screen transactions, etc.

You can also conduct tests for combinations of various OS versions and resolutions without additional human resources. Designing test architecture for automation is also important.

Implement the Strategy

We’ve been developing UI tests since 2014 to support re-engineering. Before, we had many manual tests and non-automated UI tests. But currently we have many UI tests, which cover around 80% screen transaction and support manual testing.

Our architecture keeps the independencies between end user scenario and the concrete implementations depends on iOS framework. Thus we can decrease maintenance costs for scenario side and iOS side.

Scenarios With Data-driven Testing

I implemented scenarios for data-driven testing style with Turnip. Turnip arose to describe tests with natural language. This scenario is close to the end-user. These are the steps and wrappers to convert the natural language to programming languages. Here, it’s done in Ruby.

step "Conduct tests with :device device" do |device|
 start_device(device)
end

def start_device(device)
  driver_start_with(desired_capabilities: des_app_caps, server_caps: des_server_caps)
  set_location
  close_initial_information(device)
end

Tips

1. Reduce dependency from the internal product

It’s important to describe test scenarios independent of internal logic from the production code. For example, this find_element must try to find out elements with xpath.

find_element :xpath, "//UIAApplication[1]/UIAWindow[1]/UIATableView[1]/UIATableCell[1]"

Xpath indicates a view hierarchy. But finding elements will easily be broken if the OS version is updated, as the pair strongly depends on the iOS frameworks logic.

2. Don’t conduct tests for all boundaries in UI

For example when testing validations for the text field, because UI tests are too slow, if you implement all boundaries there, it will be inefficient. It is better to move tests from UI level to method level and refactor there.

Checking Other Results

Test automation is more than just a pass/fail of a result, it can include different tools that provide feedback to different teams. For example, we also implement an image diff to compare the app with the design, and provide feedback to the designers. Designers can know the difference between the previous version and the new version easily.

Introducing Swift to Our App

We introduced Swift into the app a few months ago. Currently, nearly 30% of the code is already implemented in Swift for our application.

The current UI tests are ample for current development processes, but as our team becomes bigger, I will introduce alternative tools such as XEY test for Earl Gray for UI testing. This will be used in conjection with Appium, as its responsibility and test frameworkout lifecycle is different.


Kazuaki Matsuo

Kazuaki Matsuo

Kazuaki Matsuo is a Quality/Test Engineer at Cookpad, one of the most popular recipe sharing apps in the world.

Edited by Will Ha