// // 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() .alert(Strings.Views.Diagnostics.ReportIssue.title, isPresented: $isPresentingUnableToEmail) { Button(Strings.Global.Nouns.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: .app(title: Strings.Views.Diagnostics.Rows.app)) navLink(Strings.Views.Diagnostics.Rows.tunnel, to: .tunnel(title: Strings.Views.Diagnostics.Rows.tunnel, url: nil)) LogsPrivateDataToggle() } .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: .tunnel(title: dateString, url: item.url)) } removeAction: { removeTunnelLog(at: item.url) } } func navLink(_ title: String, to value: DebugLogRoute) -> 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() }