passepartout-apple/Passepartout/Library/Sources/AppUIMain/Views/Diagnostics/DiagnosticsView.swift
Davide 2790a2aac2
Merge AppLibrary into CommonLibrary ()
CommonLibrary barely used as standalone, almost always together with
AppLibrary.
2024-11-03 13:16:13 +01:00

204 lines
6.4 KiB
Swift

//
// DiagnosticsView.swift
// Passepartout
//
// Created by Davide De Rosa on 8/24/24.
// Copyright (c) 2024 Davide De Rosa. All rights reserved.
//
// https://github.com/passepartoutvpn
//
// This file is part of Passepartout.
//
// Passepartout is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// Passepartout is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with Passepartout. If not, see <http://www.gnu.org/licenses/>.
//
import CommonLibrary
import PassepartoutKit
import SwiftUI
struct DiagnosticsView: View {
struct LogEntry: Identifiable, Equatable {
let date: Date
let url: URL
var id: Date {
date
}
}
@EnvironmentObject
private var theme: Theme
@EnvironmentObject
private var iapManager: IAPManager
@AppStorage(AppPreference.logsPrivateData.key, store: .appGroup)
private var logsPrivateData = false
let profileManager: ProfileManager
let tunnel: ExtendedTunnel
var availableTunnelLogs: () async -> [LogEntry] = {
await Task.detached {
PassepartoutConfiguration.shared.availableLogs(at: BundleConfiguration.urlForTunnelLog)
.sorted {
$0.key > $1.key
}
.map {
LogEntry(date: $0, url: $1)
}
}.value
}
@State
private var tunnelLogs: [LogEntry] = []
@State
var isPresentingUnableToEmail = false
private let dateFormatter: DateFormatter = {
let df = DateFormatter()
df.dateFormat = Constants.shared.formats.timestamp
return df
}()
var body: some View {
Form {
liveLogSection
openVPNSection
tunnelLogsSection
if iapManager.isEligibleForFeedback() {
reportIssueSection
}
}
.task {
tunnelLogs = await availableTunnelLogs()
}
.themeForm()
.navigationTitle(Strings.Views.Diagnostics.title)
.alert(Strings.Views.Diagnostics.ReportIssue.title, isPresented: $isPresentingUnableToEmail) {
Button(Strings.Global.ok, role: .cancel) {
isPresentingUnableToEmail = false
}
} message: {
Text(Strings.Views.Diagnostics.Alerts.ReportIssue.email)
}
}
}
private extension DiagnosticsView {
var liveLogSection: some View {
Group {
navLink(Strings.Views.Diagnostics.Rows.app, to: .appDebugLog(title: Strings.Views.Diagnostics.Rows.app))
navLink(Strings.Views.Diagnostics.Rows.tunnel, to: .tunnelDebugLog(title: Strings.Views.Diagnostics.Rows.tunnel, url: nil))
Toggle(Strings.Views.Diagnostics.Rows.includePrivateData, isOn: $logsPrivateData)
.onChange(of: logsPrivateData) {
PassepartoutConfiguration.shared.logsAddresses = $0
PassepartoutConfiguration.shared.logsModules = $0
}
}
.themeSection(header: Strings.Views.Diagnostics.Sections.live)
}
var tunnelLogsSection: some View {
Group {
Button(Strings.Views.Diagnostics.Rows.removeTunnelLogs) {
withAnimation(theme.animation(for: .diagnostics), removeTunnelLogs)
}
.disabled(tunnelLogs.isEmpty)
ForEach(tunnelLogs, id: \.date, content: logView)
.onDelete(perform: removeTunnelLogs)
}
.themeSection(header: Strings.Views.Diagnostics.Sections.tunnel)
.themeAnimation(on: tunnelLogs, category: .diagnostics)
}
var openVPNSection: some View {
tunnel.value(forKey: TunnelEnvironmentKeys.OpenVPN.serverConfiguration)
.map { cfg in
Group {
NavigationLink(Strings.Views.Diagnostics.Openvpn.Rows.serverConfiguration) {
OpenVPNView(serverConfiguration: cfg)
.navigationTitle(Strings.Views.Diagnostics.Openvpn.Rows.serverConfiguration)
}
}
.themeSection(header: Strings.Unlocalized.openVPN)
}
}
var reportIssueSection: some View {
Section {
ReportIssueButton(
profileManager: profileManager,
tunnel: tunnel,
title: Strings.Views.Diagnostics.ReportIssue.title,
purchasedProducts: iapManager.purchasedProducts,
isUnableToEmail: $isPresentingUnableToEmail
)
}
}
func logView(for item: LogEntry) -> some View {
ThemeRemovableItemRow(isEditing: true) {
let dateString = dateFormatter.string(from: item.date)
navLink(dateString, to: .tunnelDebugLog(title: dateString, url: item.url))
} removeAction: {
removeTunnelLog(at: item.url)
}
}
func navLink(_ title: String, to value: AboutRouterView.NavigationRoute) -> some View {
NavigationLink(title, value: value)
}
}
private extension DiagnosticsView {
func removeTunnelLog(at url: URL) {
guard let firstIndex = tunnelLogs.firstIndex(where: { $0.url == url }) else {
return
}
try? FileManager.default.removeItem(at: url)
tunnelLogs.remove(at: firstIndex)
}
func removeTunnelLogs(at offsets: IndexSet) {
offsets.forEach {
try? FileManager.default.removeItem(at: tunnelLogs[$0].url)
}
tunnelLogs.remove(atOffsets: offsets)
}
func removeTunnelLogs() {
tunnelLogs.forEach {
try? FileManager.default.removeItem(at: $0.url)
}
tunnelLogs.removeAll()
}
}
#Preview {
DiagnosticsView(profileManager: .mock, tunnel: .mock) {
[
.init(date: Date(), url: URL(string: "http://one.com")!),
.init(date: Date().addingTimeInterval(-60), url: URL(string: "http://two.com")!),
.init(date: Date().addingTimeInterval(-600), url: URL(string: "http://three.com")!)
]
}
.withMockEnvironment()
}