Реализация сбора диагностических логов в мобильном приложении
Пользователь сообщает о проблеме — но воспроизвести её на другом устройстве не получается. Без диагностических логов остаётся угадывать. Правильно настроенная система логирования позволяет получить полный контекст прямо из приложения пользователя: последовательность действий, сетевые запросы, состояние памяти.
Архитектура логирования
In-memory кольцевой буфер
Постоянная запись в файл на каждый лог-вызов — дорогостоящая операция. Вместо этого держим кольцевой буфер в памяти и сбрасываем на диск только при сборе диагностики или крэше:
// Android: кольцевой буфер логов
class LogBuffer(private val capacity: Int = 500) {
private val buffer = ArrayDeque<LogEntry>(capacity)
@Synchronized
fun add(level: LogLevel, tag: String, message: String) {
if (buffer.size >= capacity) buffer.removeFirst()
buffer.addLast(LogEntry(
timestamp = System.currentTimeMillis(),
level = level,
tag = tag,
message = message
))
}
@Synchronized
fun getLast(count: Int): List<LogEntry> =
buffer.takeLast(minOf(count, buffer.size))
}
500 записей — достаточно, чтобы восстановить последние несколько минут работы приложения. Больше — обычно избыточно и занимает заметный объём памяти.
Timber tree для Android
class DiagnosticTree(private val buffer: LogBuffer) : Timber.Tree() {
override fun log(priority: Int, tag: String?, message: String, t: Throwable?) {
val level = when (priority) {
Log.DEBUG -> LogLevel.DEBUG
Log.INFO -> LogLevel.INFO
Log.WARN -> LogLevel.WARN
Log.ERROR -> LogLevel.ERROR
else -> LogLevel.VERBOSE
}
buffer.add(level, tag ?: "App", message)
// В Debug-сборке дополнительно пишем в Android Logcat
if (BuildConfig.DEBUG) super.log(priority, tag, message, t)
}
}
// Application.onCreate()
Timber.plant(DiagnosticTree(logBuffer))
Timber.plant(if (BuildConfig.DEBUG) Timber.DebugTree() else SilentTree())
iOS — OSLog + in-memory buffer
// Кастомный Logger с буфером
class DiagnosticLogger {
private let osLog = Logger(subsystem: "com.example.app", category: "diagnostic")
private var buffer: [LogEntry] = []
private let maxEntries = 500
private let queue = DispatchQueue(label: "logger", qos: .utility)
func log(_ message: String, level: LogLevel = .info, file: String = #file, line: Int = #line) {
let entry = LogEntry(
timestamp: Date(),
level: level,
message: message,
location: "\(URL(fileURLWithPath: file).lastPathComponent):\(line)"
)
queue.async { [weak self] in
guard let self else { return }
if self.buffer.count >= self.maxEntries {
self.buffer.removeFirst()
}
self.buffer.append(entry)
}
// OSLog для Instruments и Console.app
osLog.log(level: level.osLogType, "\(message)")
}
}
#file и #line — автоматическая привязка к месту вызова. В диагностическом отчёте видно не только «что произошло», но и «где в коде».
Сохранение в файл
При сборе диагностики сериализуем буфер в файл:
suspend fun exportDiagnosticReport(): File = withContext(Dispatchers.IO) {
val file = File(context.cacheDir, "diagnostic_${System.currentTimeMillis()}.txt")
file.bufferedWriter().use { writer ->
writer.appendLine("=== App: ${BuildConfig.VERSION_NAME} (${BuildConfig.VERSION_CODE}) ===")
writer.appendLine("=== Device: ${Build.MANUFACTURER} ${Build.MODEL}, Android ${Build.VERSION.RELEASE} ===")
writer.appendLine("=== Report generated: ${Date()} ===")
writer.appendLine()
logBuffer.getLast(500).forEach { entry ->
writer.appendLine("[${entry.levelTag}] ${entry.formattedTime} ${entry.tag}: ${entry.message}")
}
}
file
}
Обфускация чувствительных данных
Логи не должны содержать токены, пароли или данные карт. Фильтрация на уровне logger-tree:
private val sensitivePatterns = listOf(
Regex("""Bearer\s+[\w\-._~+/]+=*"""), // Authorization header
Regex("""\b\d{13,19}\b"""), // Номера карт
Regex(""""password"\s*:\s*"[^"]*"""") // JSON-поле password
)
override fun log(priority: Int, tag: String?, message: String, t: Throwable?) {
var sanitized = message
sensitivePatterns.forEach { pattern ->
sanitized = pattern.replace(sanitized, "[REDACTED]")
}
buffer.add(/* ... */, sanitized)
}
Отправка отчёта пользователем
Файл диагностики прикрепляется к форме обратной связи или отправляется по запросу поддержки:
// iOS: Share sheet для отправки файла
let diagnosticURL = try await DiagnosticLogger.shared.exportReport()
let activityVC = UIActivityViewController(
activityItems: [diagnosticURL],
applicationActivities: nil
)
present(activityVC, animated: true)
Дополнительно — возможность отправки напрямую в support-тикет через API Zendesk/Freshdesk как attachment.
Ориентиры по срокам
Реализация in-memory буфера, Timber tree / OSLog-обёртки и экспорта файла — 3–5 дней. С обфускацией чувствительных данных, интеграцией с helpdesk и UI для пользователя — до 1 недели.







