「動かないんですけど。」
iOSアプリのサポートをしたことがあるなら、この身も蓋もないフィードバックを受け取った経験があるはずです。再現手順もなく、エラーメッセージもなく、コンテキストもない――ただ恐怖の「動かない」報告だけが届き、答えよりも疑問ばかりが増えていきます。
どんなに丁寧なユーザーでも、問題の診断に必要な情報を把握していることは稀です。助けようとしてくれても、適切な詳細を提供するための技術的知識が足りないことが多いのです。このギャップが、関係者全員にとってフラストレーションの原因になります。
この記事では、このギャップを埋めるためにErrorKitに実装した2つの実践的なアプローチを紹介します:診断ログを自動収集するシンプルなフィードバックボタンと、ユーザーからの直接報告がなくてもパターンを特定できる構造化されたエラーアナリティクスです。
足りないコンテキスト問題
ユーザーが問題に遭遇したとき、診断を困難にする要因がいくつかあります:
あなたがどんな情報を必要としているかわからない
システムログに簡単にアクセスできない
正確な手順を覚えて言語化するのが難しい
複雑な問題は複数のコンポーネントが関与していることがある
間欠的な問題はオンデマンドでの再現が難しい
適切なコンテキストがなければ、デバッグは推測ゲームになります。正しい情報があれば数分で解決できる問題に、何時間も費やすことになりかねません。
解決策1:ログ付きフィードバックボタン
最初の解決策は、ユーザーが完全な情報を送るのを驚くほど簡単にすることです。ErrorKitは、自動ログ収集付きのメールコンポーザーを追加するSwiftUIモディファイアを提供します:
struct ContentView: View {
@State private var showMailComposer = false
var body: some View {
VStack {
// Your app content
Button("Report a Problem") {
showMailComposer = true
}
.mailComposer(
isPresented: $showMailComposer,
recipient: "[email protected]",
subject: "YourApp Bug Report",
messageBody: """
Please describe what happened:
----------------------------------
Device: \(UIDevice.current.model)
iOS: \(UIDevice.current.systemVersion)
App version: \(Bundle.main.infoDictionary?["CFBundleShortVersionString"] as? String ?? "Unknown")
""",
attachments: [
try? ErrorKit.logAttachment(ofLast: .minutes(30))
]
)
}
}
}これにより、シンプルな「問題を報告」ボタンが作成され、以下のことが行われます:
事前入力済みのメールコンポーザーを開く
デバイスとアプリの情報を含める
最近のシステムログを自動的に添付する
ユーザーが問題を説明するスペースを提供する
ログ添付がここでの秘密兵器です。ユーザーが問題発生後にこのボタンをタップすると、問題が起きた前後にアプリの中や周辺で何が起きていたかの包括的な情報が手に入ります。
Apple の統合ログシステムの活用
ErrorKitはAppleの統合ログシステム(OSLog/Logger)を使って診断情報を収集します。構造化ログをまだ使っていない方向けに、簡単に紹介します:
import OSLog
// Create loggers
let logger = Logger()
// or with subsystem and category
let networkLogger = Logger(subsystem: "com.yourapp", category: "networking")
// Log at appropriate levels
logger.debug("Detailed connection info") // Development debugging
logger.info("User tapped submit button") // General information
logger.notice("Profile successfully loaded") // Important events
logger.error("Failed to load user data") // Errors that should be fixed
logger.fault("Database corruption detected") // System failures
// Format values and control privacy
logger.info("User \(userId, privacy: .private) logged in from \(ipAddress, privacy: .public)")統合ログシステムには print() 文に比べていくつかの利点があります:
情報をフィルタリングするためのログレベル
機密データのプライバシーコントロール
最小限のオーバーヘッドによる効率的なパフォーマンス
アプリの再起動をまたいだ永続化
包括的なログ収集
ErrorKitのアプローチの大きな利点は、自分のアプリのログだけでなく、以下からの関連ログもキャプチャすることです:
Appleの統合ログシステムを使用するサードパーティフレームワーク
アプリが連携するシステムコンポーネント(ネットワーク、ファイルシステムなど)
アプリの機能に関連するバックグラウンドプロセス
これにより、問題が発生した際にアプリの中や周辺で何が起きていたかの全体像が得られます。明示的に追加したログだけではなく、すべてが見えるのです。
ログ収集のコントロール
ログ収集は、詳細さとプライバシーのバランスを取るためにカスタマイズできます:
// Collect logs from last 30 minutes with notice level or higher (default)
try ErrorKit.logAttachment(ofLast: .minutes(30), minLevel: .notice)
// Collect logs from last hour with error level or higher (less verbose)
try ErrorKit.logAttachment(ofLast: .hours(1), minLevel: .error)
// Collect logs from last 5 minutes with debug level (very detailed)
try ErrorKit.logAttachment(ofLast: .minutes(5), minLevel: .debug)minLevel パラメータで重要度によるログのフィルタリングが可能です:
.debug:すべてのログ(非常に詳細).info:情報レベル以上のログ.notice:注目すべきイベント(デフォルト).error:エラーとフォールトのみ.fault:クリティカルなエラーのみ
これにより、診断に必要なコンテキストを確保しつつ、収集する情報量をコントロールできます。
より細かい制御のための代替メソッド
ログ処理をより細かくコントロールしたい場合、ErrorKitには追加のアプローチがあります:
ログデータの直接取得
独自のバックエンドにログを送信したり、アプリ内でログを処理するには、loggedData を使います:
let logData = try ErrorKit.loggedData(
ofLast: .minutes(10),
minLevel: .notice
)
// Use the data with your custom reporting system
analyticsService.sendLogs(data: logData)一時ファイルへのエクスポート
他のメカニズムでログを共有するには、exportLogFile を使います:
let logFileURL = try ErrorKit.exportLogFile(
ofLast: .hours(1),
minLevel: .error
)
// Share the log file
let activityVC = UIActivityViewController(
activityItems: [logFileURL],
applicationActivities: nil
)
present(activityVC, animated: true)解決策2:グルーピングIDによるスマートなエラーアナリティクス
フィードバックボタンはユーザーが気づいた問題の報告に役立ちますが、多くの問題は報告されないままです。ユーザーはエラーに遭遇しても、肩をすくめてもう一度やり直すだけで、あなたに伝えることはありません。ここでエラーアナリティクスの出番です。
ErrorKitは、エラーを自動的に追跡しインテリジェントにグルーピングするツールを提供します:
func handleError(_ error: Error) {
// Get a stable ID that ignores dynamic parameters
let groupID = ErrorKit.groupingID(for: error)
// Get the full error chain description
let errorDetails = ErrorKit.errorChainDescription(for: error)
// Send to your analytics system
Analytics.track(
event: "error_occurred",
properties: [
"error_group": groupID,
"error_details": errorDetails,
"user_id": currentUser.id
]
)
// Show appropriate UI to the user
showErrorAlert(message: ErrorKit.userFriendlyMessage(for: error))
}アプリに追加するグローバルエラー処理関数のサンプルです。
ここでの魔法は groupingID(for:) 関数にあります。この関数は、エラーの型構造とenumケースに基づいた安定した識別子を生成し、動的パラメータやローカライズされたメッセージは無視します。
つまり、同じ根本原因を持つエラーは、具体的な詳細(ファイルパスやユーザーIDなど)が異なっても、同じグルーピングIDを持つことになります:
// Both generate the same groupID: "3f9d2a"
ProfileError
└─ DatabaseError
└─ FileError.notFound(path: "/Users/john/data.db")
ProfileError
└─ DatabaseError
└─ FileError.notFound(path: "/Users/jane/backup.db")このアプローチにはいくつかの利点があります:
一般的な問題の特定:最も頻繁に発生するエラーを把握
修正の優先順位付け:影響の大きい問題に集中
解決の追跡:修正後にエラー率が減少したかモニタリング
新しい問題の検出:リリース後に新しいエラーパターンを素早く特定
ユーザーセグメントとの相関:特定のユーザーに影響するエラーがあるか確認
両方のアプローチを組み合わせて最大のインサイトを得る
自動アナリティクスとユーザー主導のフィードバックを組み合わせる強力なアプローチとして、以下のような実装が考えられます:
func handleError(_ error: Error) {
// Always track for analytics
trackErrorAnalytics(error)
// For serious or unexpected errors, prompt for feedback
if isSerious(error) {
showErrorAlert(
message: ErrorKit.userFriendlyMessage(for: error),
feedbackOption: true
)
} else {
// For minor issues, just show a message
showErrorAlert(message: ErrorKit.userFriendlyMessage(for: error))
}
}
func showErrorAlert(message: String, feedbackOption: Bool = false) {
// Implementation of an alert that optionally includes a
// "Send Feedback" button that opens the mail composer with logs
}これにより、以下のような包括的なシステムが構築されます:
すべてのエラーがアナリティクスで追跡され、全体的なパターンが把握できる
深刻なエラーではユーザーにログ付きの詳細なフィードバックを促す
ユーザーは追跡対象外の問題についても、いつでもフィードバックを開始できる
ログのベストプラクティス
ログ収集の価値を最大化するために、以下のベストプラクティスを検討してください:
1. コンテキストのある構造化ログ
何が起きていたかを理解するのに十分なコンテキストをログに含めましょう:
// Instead of:
Logger().error("Failed to load")
// Use:
Logger().error("Failed to load document \(documentId): \(ErrorKit.errorChainDescription(for: error))")2. 適切なログレベルの選択
冗長度をコントロールするために、ログレベルを戦略的に使い分けましょう:
.debug:開発中にのみ必要な開発者向けの詳細.info:通常のアプリフローの追跡.notice:ユーザーにとって重要なイベント.error:修正が必要だがコア機能を妨げない問題.fault:コア機能を破壊するクリティカルな問題
3. 機密情報の保護
プライバシーモディファイアを使ってユーザーデータを保護しましょう:
Logger().info("Processing payment for user \(userId, privacy: .private)")4. 重要なユーザーアクションの記録
エラーに至るまでのユーザー操作の軌跡(パンくずリスト)を作りましょう:
Logger().notice("User navigated to profile screen")
Logger().info("User tapped edit button")
Logger().notice("User saved profile changes")5. 重要な操作の開始と完了を記録
未完了のタスクを特定するために、重要な操作を括弧で囲みましょう:
Logger().notice("Starting data sync")
// ... sync implementation
Logger().notice("Completed data sync")サポートと開発への影響
これらのツールを導入することで、ユーザー体験と開発ワークフローの両方が変わります:
ユーザーにとって:
報告の簡素化:ワンタップでフィードバックを送信
技術的な質問が不要:ストレスのある往復コミュニケーションを回避
迅速な解決:問題をより早く診断・修正
より良い体験:問題を真剣に受け止めていることをユーザーに示せる
開発者にとって:
完全なコンテキスト:問題発生時に何が起きていたかを正確に把握
サポート時間の削減:追加情報を求めるやり取りの削減
再現性の向上:ログデータに基づくより確実な再現手順
効率的なデバッグ:エラーレポートのパターンを素早く特定
データ駆動の優先順位付け:最も一般的な問題の修正に集中
まとめ
ErrorKitのアプローチは、ユーザーの「動かない」という報告と、実際に何が起きたかを知ることの間にある、もどかしいギャップを埋めます。自動ログ収集とスマートなエラーアナリティクスの組み合わせが、実際に機能するフィードバックループを生み出すことを実感しています。
特に強力なのは、ユーザーが問題を報告するときに詳細なログが得られると同時に、ユーザーが報告しない問題もキャッチできることです。このデュアルアプローチは、アプリの問題の理解と修正の仕方を根本的に変えました。目隠しでのデバッグにうんざりしているなら、ErrorKitにはこれらのログツールとエラー処理の改善がすべて含まれています。自分自身が必要だったから作ったツールです:
ユーザーフィードバックやエラーレポートをどのように処理していますか?他に効果的なテクニックを見つけたことはありますか?ぜひソーシャルメディア(リンクは下記)で教えてください!

