[G1-Sync] Manual knowledge update
This commit is contained in:
@@ -0,0 +1,143 @@
|
||||
---
|
||||
id: native-crash-reporting
|
||||
title: Native Crash Reporting — Crashlytics / Sentry
|
||||
category: Coding
|
||||
status: draft
|
||||
source_trust_level: B
|
||||
verification_status: conceptual
|
||||
created_at: 2026-05-09
|
||||
updated_at: 2026-05-09
|
||||
tags: [native, crash, sentry, crashlytics, vibe-coding]
|
||||
tech_stack: { language: "Swift / Kotlin", applicable_to: ["iOS", "Android"] }
|
||||
applied_in: []
|
||||
aliases: [Crashlytics, Sentry, dSYM, ProGuard mapping, breadcrumbs]
|
||||
---
|
||||
|
||||
# Native Crash Reporting
|
||||
|
||||
> Prod crash 추적의 표준 = **Firebase Crashlytics 또는 Sentry**. dSYM(iOS)/mapping(Android) 업로드 필수. Symbol 없는 stack 은 쓸모없음. Breadcrumbs + user identifier + custom keys 가 디버깅 무기.
|
||||
|
||||
## 📖 핵심 개념
|
||||
- Symbolication: hex address → 함수명. dSYM/mapping 필요.
|
||||
- Non-fatal: handled exception 도 보고.
|
||||
- Breadcrumbs: 충돌 직전 N 이벤트.
|
||||
- ANR (Android): Crashlytics 자동 보고.
|
||||
|
||||
## 💻 코드 패턴
|
||||
|
||||
### iOS — Crashlytics
|
||||
```ruby
|
||||
# Podfile
|
||||
pod 'FirebaseCrashlytics'
|
||||
```
|
||||
|
||||
```swift
|
||||
import FirebaseCrashlytics
|
||||
|
||||
// 사용자 식별
|
||||
Crashlytics.crashlytics().setUserID(user.id)
|
||||
Crashlytics.crashlytics().setCustomValue(plan, forKey: "subscription")
|
||||
|
||||
// 로그 (breadcrumb)
|
||||
Crashlytics.crashlytics().log("Tapped checkout")
|
||||
|
||||
// non-fatal
|
||||
Crashlytics.crashlytics().record(error: NSError(domain: "App", code: 42))
|
||||
```
|
||||
|
||||
dSYM upload (Build Phase):
|
||||
```
|
||||
"${PODS_ROOT}/FirebaseCrashlytics/run"
|
||||
# Input Files: $(BUILT_PRODUCTS_DIR)/$(INFOPLIST_PATH), ${DWARF_DSYM_FOLDER_PATH}/${DWARF_DSYM_FILE_NAME}
|
||||
```
|
||||
|
||||
### Android — Crashlytics
|
||||
```kotlin
|
||||
// build.gradle
|
||||
plugins {
|
||||
id 'com.google.gms.google-services'
|
||||
id 'com.google.firebase.crashlytics'
|
||||
}
|
||||
|
||||
dependencies {
|
||||
implementation("com.google.firebase:firebase-crashlytics-ktx")
|
||||
implementation("com.google.firebase:firebase-analytics-ktx")
|
||||
}
|
||||
```
|
||||
|
||||
```kotlin
|
||||
val crashlytics = FirebaseCrashlytics.getInstance()
|
||||
crashlytics.setUserId(user.id)
|
||||
crashlytics.setCustomKey("subscription", "pro")
|
||||
crashlytics.log("opened CartActivity")
|
||||
|
||||
// non-fatal
|
||||
try { risky() } catch (e: Exception) { crashlytics.recordException(e) }
|
||||
```
|
||||
|
||||
ProGuard mapping 자동 업로드 (gradle plugin 처리).
|
||||
|
||||
### Sentry (cross-platform)
|
||||
```swift
|
||||
import Sentry
|
||||
|
||||
SentrySDK.start { o in
|
||||
o.dsn = "https://...@sentry.io/0"
|
||||
o.tracesSampleRate = 0.1
|
||||
o.debug = false
|
||||
}
|
||||
|
||||
SentrySDK.setUser(User(userId: user.id))
|
||||
SentrySDK.capture(error: error)
|
||||
```
|
||||
|
||||
```kotlin
|
||||
SentryAndroid.init(context) { o ->
|
||||
o.dsn = "https://...@sentry.io/0"
|
||||
o.tracesSampleRate = 0.1
|
||||
}
|
||||
|
||||
Sentry.captureException(e)
|
||||
Sentry.addBreadcrumb(Breadcrumb().apply { message = "checkout tapped" })
|
||||
```
|
||||
|
||||
### React Native — @sentry/react-native
|
||||
```ts
|
||||
import * as Sentry from '@sentry/react-native';
|
||||
|
||||
Sentry.init({ dsn: '...', tracesSampleRate: 0.1 });
|
||||
Sentry.captureException(error);
|
||||
```
|
||||
|
||||
### Custom log line strategy
|
||||
```swift
|
||||
// 단순 log 보다 key=value 로
|
||||
Crashlytics.crashlytics().log("nav route=cart user=\(user.id) cartCount=\(items.count)")
|
||||
```
|
||||
|
||||
## 🤔 의사결정 기준
|
||||
| 필요 | 추천 |
|
||||
|---|---|
|
||||
| Firebase 이미 사용 | Crashlytics (무료) |
|
||||
| Self-hosted / 풍부한 분석 | Sentry |
|
||||
| Crash + Performance 통합 | Sentry / Firebase Performance |
|
||||
| ReactNative 한 SDK | Sentry RN |
|
||||
| Symbol 자동 업로드 | Sentry CLI / Crashlytics gradle plugin |
|
||||
|
||||
## ❌ 안티패턴
|
||||
- **dSYM 업로드 누락**: 의미 없는 stack.
|
||||
- **Proguard mapping 안 올림**: Android 도 동일.
|
||||
- **PII 그대로 user log**: GDPR / CCPA 위반.
|
||||
- **breadcrumb 너무 많음**: noise. 의미 있는 이벤트만.
|
||||
- **`fatalError(...)` 무절제**: prod 사용자에 보임.
|
||||
- **catch 후 swallow**: 보고도 안 함, 디버깅 불가. 최소 record.
|
||||
- **debug build 도 SDK 활성**: noise.
|
||||
|
||||
## 🤖 LLM 활용 힌트
|
||||
- userID + custom key + log + recordException 4종.
|
||||
- dSYM/mapping 자동 업로드 setup 우선.
|
||||
|
||||
## 🔗 관련 문서
|
||||
- [[Logging_Structured_Patterns]]
|
||||
- [[Observability_Stack]]
|
||||
- [[React_Native_Crash_Tracking]]
|
||||
Reference in New Issue
Block a user