[G1-Sync] Manual knowledge update

This commit is contained in:
Antigravity Agent
2026-05-09 21:08:02 +09:00
parent f0befc887a
commit 93ec7e9056
363 changed files with 68333 additions and 64 deletions
@@ -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]]