Upgrade to Pro
— share decks privately, control downloads, hide ads and more …
Speaker Deck
Features
Speaker Deck
PRO
Sign in
Sign up for free
Search
Search
Fixing Broken Robots - Android Mutation Testing
Search
Panini
July 18, 2019
Programming
220
0
Share
Embed
Copy iframe code
Copy JS code
Copy link
Start on current slide
Fixing Broken Robots - Android Mutation Testing
Panini
July 18, 2019
More Decks by Panini
See All by Panini
Kotlin 1.5 preview
panini
1
480
Java谷園から逃げ出した話
panini
0
410
Kotlin 1.4-rc
panini
0
250
Kotlin Multiplatform
panini
2
700
build.gradle.kts
panini
2
2.5k
Kotlin Multi-platform(?)
panini
1
780
Convert Java file to Kotlin file ⌥⇧⌘K
panini
0
1.2k
Introducing Android P
panini
2
870
Display Cutout
panini
1
680
Other Decks in Programming
See All in Programming
不変条件と整合性境界—ビジネスが決める設計判断と実現パターン / Invariants and Consistency Boundaries
nrslib
13
5.5k
LLM本来の能力を解き放つサンドボックス技術とAI民主化への適用
yukukotani
3
4.3k
並列実装の現場、2ヶ月間実務でAIを使い倒したAIもPCも私も限界が近い
ming_ayami
0
130
net-httpのHTTP/2対応について
naruse
0
500
作って学ぶ、 JSX (TSX) ランタイムの基本
syumai
7
1.6k
Developing with AI Agents — Codex, Claude Code & Cowork Practical Guide
x5gtrn
PRO
0
1.3k
Creating Composable Callables in Contemporary C++
rollbear
0
150
AIだと陥りがちなJakarta EE最新技術への移行時の落とし穴と解決策
tnagao7
0
110
ローカルLLMでどこまでコードが書けるか -拡張版 / How much code can be written on a local LLM Extended
kishida
11
4.3k
The ROI of Quarkus for Spring Boot Applications
hollycummins
0
120
エンジニアと一緒にテストコードの設計と実装を改善した話
mototakatsu
0
200
ECSアプリログをFireLensでコスト削減しようとしたけど諦めた話 in Fargate×Node.js
akihisaikeda
2
4.2k
Featured
See All Featured
Efficient Content Optimization with Google Search Console & Apps Script
katarinadahlin
PRO
1
630
The untapped power of vector embeddings
frankvandijk
2
1.8k
Build your cross-platform service in a week with App Engine
jlugia
234
18k
B2B Lead Gen: Tactics, Traps & Triumph
marketingsoph
0
150
The Limits of Empathy - UXLibs8
cassininazir
1
360
Highjacked: Video Game Concept Design
rkendrick25
PRO
1
390
Embracing the Ebb and Flow
colly
88
5.1k
How to Align SEO within the Product Triangle To Get Buy-In & Support - #RIMC
aleyda
2
1.5k
How to Create Impact in a Changing Tech Landscape [PerfNow 2023]
tammyeverts
55
3.4k
Marketing Yourself as an Engineer | Alaka | Gurzu
gurzu
0
240
Six Lessons from altMBA
skipperchong
29
4.3k
<Decoding/> the Language of Devs - We Love SEO 2024
nikkihalliwell
1
250
Transcript
Fixing broken robots - Android Mutation Testing DroidConSG 2019 1/45
About me Matthew Vern Twitter Github Mercari, Inc Software Engineer
(Android) @panini_ja panpanini 2/45
Client Engineer Solving problems for our customers Shipping features Improve
existing functionality My job 3/45
A non-shipped feature doesn't provide benefit Ship features as quick
as possible My job 4/45
A shipped, broken feature doesn't provide benefit Ship quality features
as quick as possible My job 5/45
Maintaining quality 6/45
QA Maintaining quality 6/45
QA Code Review Maintaining quality 6/45
QA Code Review Tests Maintaining quality 6/45
How do we know our tests are providing quality Maintaining
quality 7/45
How do we know our tests are providing quality Use
coverage to make sure that our tests are calling production code Maintaining quality 7/45
How do we know our tests are providing quality Use
coverage to make sure that our tests are calling production code changes introduced will not break existing code Maintaining quality 7/45
How do we know our tests are providing quality Use
coverage to make sure that our tests are calling production code changes introduced will not break existing code new code does what it says on the tin Maintaining quality 7/45
Who watches the watchmen? 8/45
How do we know that our tests are quality? Maintaining
quality 9/45
What are tests 10/45
asserting that our assumptions about a piece of code are
correct binary assertions of code correctness What are tests 11/45
Lets fail some tests 12/45
Unit tests assert code behaviour change code behaviour tests fail
???? profit Lets fail some tests 13/45
Mutation testing 14/45
proposed by Richard Lipton in 1971 computationally expensive, not a
viable testing solution until recently Mutation testing 15/45
1. Create a mutant 2. Run test suite 3. Confirm
if mutant was detected or not 4. Repeat Mutation testing steps 16/45
A mutant is a biological entity which has undergone a
change in its genetic structure. What is a mutant? 17/45
A mutant is a code block which has undergone a
change in its structure. What is a mutant? 18/45
class SessionController( private val sessions: MutableList<Session> ) : EpoxyController() {
fun setSessions(sessions: List<Session>) {} override fun buildModels() {} fun generateModels(sessions: List<Session>): List<SessionModel> {} } Creating mutations 19/45
Creating mutations fun setSessions(sessions: List<Session>) { this.sessions.clear() this.sessions.addAll(sessions) requestModelBuild() }
20/45
Creating mutations override fun buildModels() { generateModels(sessions) .forEach { it.addTo(this)
} } 21/45
Creating mutations fun generateModels(sessions: List<Session>): List<SessionModel> { return sessions .map
{ session SessionModel_() .title(session.title) .imageUrl( if (session.speaker.profileImage "") { session.speaker.profileImage } else { null } ) } } 22/45
Creating mutations 23/45
Competent Programmer Hypothesis Creating mutations 23/45
Competent Programmer Hypothesis Coupling Effect Creating mutations 23/45
Replaces relational operators with boundary counterpart Original Original Mutated Mutated
< <= <= < > >= >= > Conditionals boundary 24/45
Conditionals boundary original if (currentTime < startTime) { do something
} mutated if (currentTime startTime) { do something } 25/45
Negates conditional checks Original Original Mutated Mutated == != !=
== <= > > <= Negate Conditionals 26/45
Negate Conditionals original fun buildModels() { SessionModel_() .title(session.title) .imageUrl( if
(session.speaker.profileImage "") { session.speaker.profileImage } else { null } ) } mutated fun buildModels() { SessionModel_() .title(session.title) .imageUrl( if (session.speaker.profileImage "") { session.speaker.profileImage } else { null } ) } 27/45
removes void method calls Remove void calls 28/45
Remove void calls original fun setSessions(sessions: List<Session>) { this.sessions.clear() this.sessions.addAll(sessions)
requestModelBuild() } mutated fun setSessions(sessions: List<Session>) { this.sessions.clear() this.sessions.addAll(sessions) } 29/45
So what? 30/45
Code coverage, but better Why mutation testing 31/45
1. Introduce a fault into production code 2. Use code
coverage to determine which tests to run 3. Run tests 4. Confirm if fault was detected or not 5. Repeat Mutation testing The better way 32/45
That's a lot of work you expect us to do
there bud 33/45
Pitest 34/45
mutation testing system mutants stored in memory outputs pretty reports
Gradle plugin Pitest pitest.org 35/45
apply plugin: pitest generates pitest<Variant> tasks Gradle plugin szpak/gradle-pitest-plugin 36/45
37/45
written by Karol Wrótniak, forked from szpak/gradle-pitest-plugin works with Android
projects has some Android specific helpers (eg: generating mockable Android jar) Android Gradle plugin koral--/gradle-pitest-plugin 38/45
plugins { id("pl.droidsonroids.pitest") } pitest { excludeMockableAndroidJar = false targetClasses
= setOf("jp.co.panpanini.mypackage.*") outputFormats = setOf("XML", "HTML") } Android Gradle plugin 39/45
id("pl.droidsonroids.pitest") plugins { } pitest { excludeMockableAndroidJar = false targetClasses
= setOf("jp.co.panpanini.mypackage.*") outputFormats = setOf("XML", "HTML") } Android Gradle plugin 39/45
pitest { excludeMockableAndroidJar = false targetClasses = setOf("jp.co.panpanini.mypackage.*") outputFormats =
setOf("XML", "HTML") } plugins { id("pl.droidsonroids.pitest") } Android Gradle plugin 39/45
excludeMockableAndroidJar = false plugins { id("pl.droidsonroids.pitest") } pitest { targetClasses
= setOf("jp.co.panpanini.mypackage.*") outputFormats = setOf("XML", "HTML") } Android Gradle plugin 39/45
targetClasses = setOf("jp.co.panpanini.mypackage.*") plugins { id("pl.droidsonroids.pitest") } pitest { excludeMockableAndroidJar
= false outputFormats = setOf("XML", "HTML") } Android Gradle plugin 39/45
outputFormats = setOf("XML", "HTML") plugins { id("pl.droidsonroids.pitest") } pitest {
excludeMockableAndroidJar = false targetClasses = setOf("jp.co.panpanini.mypackage.*") } Android Gradle plugin 39/45
Demo 40/45
Pitest tips & tricks 41/45
MutationInterceptor Removes mutants for Kotlin generated code Pitest kotlin pitest/pitest-kotlin
42/45
Run PITest on Unit tests only 43/45
Run PITest on CI 44/45
Mutation Testing panpanini/mutation_testing Github: panpanini Twitter: panini_ja 45/45