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
新規アプリの単体テスト戦略 / unit_tests_strategy_of_new_app
Search
uhooi
October 28, 2022
Programming
3
940
新規アプリの単体テスト戦略 / unit_tests_strategy_of_new_app
iOS Test Online
https://testonline.connpass.com/event/261910/
uhooi
October 28, 2022
Tweet
Share
More Decks by uhooi
See All by uhooi
snacks.nvim内のセットアップ不要なプラグインを紹介 / introduce_snacks_nvim
uhooi
3
660
Develop iOS apps with Neovim / vimconf_2024
uhooi
2
620
暴走のウホーレン 〜想いってのはvimrcにしないと伝わらないんだぜ〜 / iosdc_japan_2024
uhooi
1
530
SwiftLintのカスタムルールを作るための正規表現入門 / swiftlint_regex
uhooi
1
150
最古の関数型言語「Lisp」ことはじめ / lisp_in_kamiyama
uhooi
1
330
会社PCでrm -rf ~した話 / rm_rf_choron
uhooi
18
24k
クソアプリを作ってみた💩 / kusojaku
uhooi
0
620
Apple提供アプリのナビゲーションバーアイテムクイズ / 20231110-hig
uhooi
2
290
個人開発のiOSアプリでUI/UXを標準に寄せてみた / 20230919_orochi
uhooi
1
980
Other Decks in Programming
See All in Programming
CDK引数設計道場100本ノック
badmintoncryer
2
560
React は次の10年を生き残れるか:3つのトレンドから考える
oukayuka
40
14k
SQLアンチパターン第2版 データベースプログラミングで陥りがちな失敗とその対策 / Intro to SQL Antipatterns 2nd
twada
PRO
26
8k
11年かかって やっとVibe Codingに 時代が追いつきましたね
yimajo
0
150
顧客の画像データをテラバイト単位で配信する 画像サーバを WebP にした際に起こった課題と その対応策 ~継続的な取り組みを添えて~
takutakahashi
4
1.4k
状態遷移図を書こう / Sequence Chart vs State Diagram
orgachem
PRO
3
250
「App Intent」よくわからんけどすごい!
rinngo0302
1
120
新メンバーも今日から大活躍!SREが支えるスケールし続ける組織のオンボーディング
honmarkhunt
5
9.3k
Googleの新しいコーディングAIエージェントJulesを使ってみた
tonionagauzzi
0
120
Claude Code で Astro blog を Pages から Workers へ移行してみた
codehex
0
150
AIともっと楽するE2Eテスト
myohei
9
3.1k
テスターからテストエンジニアへ ~新米テストエンジニアが歩んだ9ヶ月振り返り~
non0113
2
240
Featured
See All Featured
Code Reviewing Like a Champion
maltzj
524
40k
Site-Speed That Sticks
csswizardry
10
720
Building a Scalable Design System with Sketch
lauravandoore
462
33k
Stop Working from a Prison Cell
hatefulcrawdad
271
21k
Reflections from 52 weeks, 52 projects
jeffersonlam
351
21k
Refactoring Trust on Your Teams (GOTO; Chicago 2020)
rmw
34
3.1k
Fantastic passwords and where to find them - at NoRuKo
philnash
51
3.3k
Making Projects Easy
brettharned
116
6.3k
Facilitating Awesome Meetings
lara
54
6.5k
Visualization
eitanlees
146
16k
Building Better People: How to give real-time feedback that sticks.
wjessup
367
19k
Building Flexible Design Systems
yeseniaperezcruz
328
39k
Transcript
新規アプリの 単体テスト戦略 2022.10.28(Fri) iOS Test Online @the_uhooi
iOS app developer @uhooi @uhooi @the_uhooi
入社前の会話 新規アプリ作って!
入社前の会話 新規アプリ作って! はい!
ウホーイ入社 ウホーイ 入社
・UI: フル SwiftUI ・リアクティブ: Combine ・非同期処理: Swift Concurrency (async/await) ・アーキテクチャ:
MVVM ・その他: SwiftPM によるマルチモジュール、KMM ※開発中なので変更される可能性が大いにあります。 新規アプリの技術選定
最初に考えた単体テストの戦略
できる限り にする 単体テストが不要な設計 最初に考えた単体テストの戦略
できる限り にする 単体テストが不要な設計 必要な にする 単体テストを書きやすい設計 最初に考えた単体テストの戦略
01 単体テストが な 設計にする 不要
アーキテクチャ(MVVM) View View Model UseCase KMM
単体テストを書く層を明文化する View View Model UseCase KMM : 単体テストを書かない : 必要に応じて単体テストを書く
: 単体テストを書く
View 層の単体テストを書かない理由 UI/UX を確認するため、 必ず から 手動でテストを行う
どうしたら View 層から単体テストをなくせる? View View Model UseCase KMM : 単体テストを書かない
: 必要に応じて単体テストを書く : 単体テストを書く
ロジックを減らせば単体テストも減らせる View View Model UseCase KMM 少なくしたい 多くする : 単体テストを書かない
: 必要に応じて単体テストを書く : 単体テストを書く View 層からロジックを なくすのが理想 ロジック
ロジックを減らすには? ロジックを減らすには?
分岐をなくせばロジックを減らせる ロジックを減らすには? 分岐 をなくす (if 文や switch 文)
View に分岐がある(Before) // Before // View 層 if viewModel.uiState.isLoading {
Text("ロード中です。") } else { Text("ロードが完了しました。") }
View から分岐をなくす(After) // After // View 層 Text(viewModel.uiState.loadingText) // ViewModel
層 var loadingText: String { isLoading ? "ロード中です。" : "ロードが完了しました。" }
View から分岐をなくす(After) // After // View 層 Text(viewModel.uiState.loadingText) // ViewModel
層 var loadingText: String { isLoading ? "ロード中です。" : "ロードが完了しました。" } View 層の分岐を ViewModel 層に寄せる
それでも View から分岐は完全にはなくせない // View 層 struct MonstersView: View {
var body: some View { if viewModel.uiState.monsters.isEmpty { EmptyMonsterListView() } else { MonsterListView(monsters: viewModel.uiState.monsters) } } }
それでも View から分岐は完全にはなくせない // View 層 struct MonstersView: View {
var body: some View { if viewModel.uiState.monsters.isEmpty { EmptyMonsterListView() } else { MonsterListView(monsters: viewModel.uiState.monsters) } } } View の出し分けなどの 分岐はなくせない
それでも View から分岐は完全にはなくせない // View 層 struct MonstersView: View {
var body: some View { if viewModel.uiState.monsters.isEmpty { EmptyMonsterListView() } else { MonsterListView(monsters: viewModel.uiState.monsters) } } } View の出し分けなどの 分岐はなくせない monsters が空のときの UI の確認が手間
そこでプレビュー そこでプレビュー
Xcode 上で SwiftUI の View を確認できる
「UI を確認するだけの手動テスト」をなくせる プレビューでは ・端末のサイズ ・ライト / ダークモード ・データの最小 / 最大値
・エラー時の View etc... 様々な状態をかんたんに作り出せる
02 単体テストを 設計にする 書きやすい
ViewModel 層以下の単体テストを書きやすくしたい View View Model UseCase KMM : 単体テストを書かない :
必要に応じて単体テストを書く : 単体テストを書く
UI フレームワークを View 層のみでインポートする View View Model UseCase KMM SwiftUI
Combine : 単体テストを書かない : 必要に応じて単体テストを書く : 単体テストを書く View 層のみで UI を完結させる
UI フレームワークを View 層のみでインポートする View View Model UseCase KMM SwiftUI
Combine : 単体テストを書かない : 必要に応じて単体テストを書く : 単体テストを書く ViewModel 層以下がロジックに集中でき、 単体テストが書きやすくなる
Task があると単体テストしにくい(Before) // Before // ViewModel 層 struct FooViewModel: ObservableObject
{ func didTapLoginButton() { Task { await login() } } }
Task があると単体テストしにくい(Before) // Before // ViewModel 層 struct FooViewModel: ObservableObject
{ func didTapLoginButton() { Task { await login() } } } ログイン処理を待たずに テストが終了してしまう
ViewModel 層のメソッドを非同期にする(After) // After // ViewModel 層 struct FooViewModel: ObservableObject
{ func didTapLoginButton() async { await login() } }
ViewModel 層のメソッドを非同期にする(After) // After // ViewModel 層 struct FooViewModel: ObservableObject
{ func didTapLoginButton() async { await login() } } 非同期にすると 並行処理もしやすくなる
Task を View 層に書く(After) // After // View 層 struct
FooView: View { // ... Task { await viewModel.didTapLoginButton() } // ... }
ちなみに…
ちなみに… まだ単体テストを書いていません これから書きます!
03 まとめ
まとめ 設計を工夫することで、 、 できる 不安定な単体テストを減らし 安定した単体テストを書きやすく
まとめ View View Model UseCase KMM UI ロジック Task 単体テストしにくい
単体テストしやすい
まとめ View View Model UseCase KMM 単体テストしにくい・しやすいものを 分離し、単体テストする層を明確にする UI ロジック
Task 単体テストしにくい 単体テストしやすい
まとめ View View Model UseCase KMM UI ロジック Task 単体テストを書かない層(View
層)は プレビューや手動テストで品質を担保する 単体テストしにくい 単体テストしやすい