Skip to content

Commit

Permalink
Fix brave/brave-ios#8007: Add additional details to webcompat report (b…
Browse files Browse the repository at this point in the history
  • Loading branch information
cuba authored Oct 3, 2023
1 parent d9c4513 commit a3aea32
Show file tree
Hide file tree
Showing 4 changed files with 186 additions and 43 deletions.
17 changes: 16 additions & 1 deletion Sources/Brave/Frontend/Shields/ShieldsViewController.swift
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@ import BraveUI
import UIKit
import Growth
import BraveCore
import BraveVPN

/// Displays shield settings and shield stats for a given URL
class ShieldsViewController: UIViewController, PopoverContentComponent {
Expand Down Expand Up @@ -326,7 +327,21 @@ class ShieldsViewController: UIViewController, PopoverContentComponent {
@objc private func tappedSubmitReportingButton() {
if let url = url {
Task { @MainActor in
await WebcompatReporter.reportIssue(on: url)
let domain = Domain.getOrCreate(forUrl: url, persistent: !tab.isPrivate)

let report = WebcompatReporter.Report(
fullUrl: url,
areShieldsEnabled: !domain.areAllShieldsOff,
adBlockLevel: domain.blockAdsAndTrackingLevel,
fingerprintProtectionLevel: domain.finterprintProtectionLevel,
adBlockListTitles: FilterListStorage.shared.filterLists.compactMap({ filterList -> String? in
guard filterList.isEnabled else { return nil }
return filterList.entry.title
}),
isVPNEnabled: BraveVPN.isConnected
)

await WebcompatReporter.send(report: report)
try await Task.sleep(nanoseconds: NSEC_PER_SEC * 2)
guard !self.isBeingDismissed else { return }
self.dismiss(animated: true)
Expand Down
193 changes: 155 additions & 38 deletions Sources/BraveShields/WebcompatReporter.swift
Original file line number Diff line number Diff line change
Expand Up @@ -8,77 +8,194 @@ import Shared
import os.log

public class WebcompatReporter {
private struct BaseURL {
static let staging = "laptop-updates.bravesoftware.com"
static let prod = "laptop-updates.brave.com"
static let log = Logger(subsystem: Bundle.main.bundleIdentifier!, category: "WebcompatReporter")

/// The raw values of the web-report.
public struct Report {
/// The URL of the broken site.
/// - Note: This is the full url and will be used to extract all relevant information
let fullUrl: URL
/// Any user input details
let additionalDetails: String?
/// Any user input contact details that may be provided
let contactInfo: String?
/// A bool indicating if shields are enabled for that site
let areShieldsEnabled: Bool
/// The level of adblocking currently set for the page
let adBlockLevel: ShieldLevel
/// The level of fingerprinting protection currently set for this page
let fingerprintProtectionLevel: ShieldLevel
/// Titles of all enabled filter lists
let adBlockListTitles: [String]
/// If VPN is currently enabled
let isVPNEnabled: Bool

var domain: String? {
return fullUrl.normalizedHost() != nil ? fullUrl.domainURL.absoluteString : fullUrl.baseDomain
}

var cleanedURL: URL? {
var components = URLComponents(url: fullUrl, resolvingAgainstBaseURL: false)
components?.fragment = nil
components?.queryItems = nil
return components?.url
}

public init(
fullUrl: URL, additionalDetails: String? = nil, contactInfo: String? = nil,
areShieldsEnabled: Bool, adBlockLevel: ShieldLevel, fingerprintProtectionLevel: ShieldLevel,
adBlockListTitles: [String], isVPNEnabled: Bool
) {
self.fullUrl = fullUrl
self.additionalDetails = additionalDetails
self.contactInfo = contactInfo
self.areShieldsEnabled = areShieldsEnabled
self.adBlockLevel = adBlockLevel
self.fingerprintProtectionLevel = fingerprintProtectionLevel
self.adBlockListTitles = adBlockListTitles
self.isVPNEnabled = isVPNEnabled
}
}

private struct Payload: Encodable {
let report: Report
let apiKey: String?
let languageCode: String?

enum CodingKeys: String, CodingKey {
case url
case domain
case additionalDetails
case contactInfo
case apiKey = "api_key"

case fpBlockSetting
case adBlockSetting
case adBlockLists
case shieldsEnabled
case languages
case languageFarblingEnabled
case braveVPNEnabled
}

public func encode(to encoder: Encoder) throws {
// We want to ensure that the URL _can_ be normalized, since `domainURL` will return itself
// (the full URL) if the URL can't be normalized. If the URL can't be normalized, send only
// the base domain without scheme.
guard let domain = report.domain else {
throw EncodingError.invalidValue(CodingKeys.domain, EncodingError.Context(
codingPath: encoder.codingPath, debugDescription: "Cannot extract `domain` from url"
))
}

guard let apiKey = apiKey else {
throw EncodingError.invalidValue(CodingKeys.apiKey, EncodingError.Context(
codingPath: encoder.codingPath, debugDescription: "Missing api_key"
))
}

guard let cleanedURL = report.cleanedURL else {
throw EncodingError.invalidValue(CodingKeys.domain, EncodingError.Context(
codingPath: encoder.codingPath, debugDescription: "Cannot strip fragments or query params"
))
}

var container: KeyedEncodingContainer<CodingKeys> = encoder.container(keyedBy: CodingKeys.self)
try container.encode(cleanedURL.absoluteString, forKey: .url)
try container.encode(domain, forKey: .domain)
try container.encodeIfPresent(report.additionalDetails, forKey: .additionalDetails)
try container.encodeIfPresent(report.contactInfo, forKey: .contactInfo)
try container.encodeIfPresent(languageCode, forKey: .languages)
try container.encode(true, forKey: .languageFarblingEnabled) // This is always enabled in iOS web-kit
try container.encode(report.areShieldsEnabled, forKey: .shieldsEnabled)
try container.encode(report.isVPNEnabled, forKey: .braveVPNEnabled)
try container.encode(report.adBlockListTitles.joined(separator: ","), forKey: .adBlockLists)
try container.encode(report.fingerprintProtectionLevel.reportLabel, forKey: .fpBlockSetting)
try container.encode(report.adBlockLevel.reportLabel, forKey: .adBlockSetting)
try container.encode(apiKey, forKey: .apiKey)
}
}

private static var baseHost: String {
if AppConstants.buildChannel == .debug {
return "laptop-updates.bravesoftware.com"
} else {
return "laptop-updates.brave.com"
}
}

private static let apiKeyPlistKey = "API_KEY"
private static let version = "1"

/// A custom user agent to send along with reports
public static var userAgent: String?

/// Get the user's language code
private static var currentLanguageCode: String? {
if #available(iOS 16, *) {
return Locale.current.language.languageCode?.identifier
} else {
return Locale.current.languageCode
}
}

/// Report a webcompat issue on a given website
///
/// - Returns: A deferred boolean on whether or not it reported successfully (default queue: main)
@discardableResult
public static func reportIssue(on url: URL) async -> Bool {
let baseURL = AppConstants.buildChannel == .debug ? BaseURL.staging : BaseURL.prod
public static func send(report: Report) async -> Bool {
let apiKey = (Bundle.main.infoDictionary?[apiKeyPlistKey] as? String)?.trimmingCharacters(in: .whitespacesAndNewlines)
let payload = Payload(report: report, apiKey: apiKey, languageCode: currentLanguageCode)

var components = URLComponents()
components.scheme = "https"
components.host = baseURL
components.host = baseHost
components.path = "/\(version)/webcompat"

guard let baseDomain = url.baseDomain,
let key = apiKey,
let endpoint = components.url
else {
Logger.module.error("Failed to setup webcompat request")
guard let endpoint = components.url else {
Self.log.error("Failed to setup webcompat request")
return false
}

// We want to ensure that the URL _can_ be normalized, since `domainURL` will return itself
// (the full URL) if the URL can't be normalized. If the URL can't be normalized, send only
// the base domain without scheme.
let domain = url.normalizedHost() != nil ? url.domainURL.absoluteString : baseDomain

let payload = [
"domain": domain,
"api_key": key,
]

do {
let encoder = JSONEncoder()
var request = URLRequest(url: endpoint)
request.httpMethod = "POST"
request.httpBody = try JSONSerialization.data(withJSONObject: payload, options: [])
request.addValue("application/json", forHTTPHeaderField: "Content-Type")
request.httpBody = try encoder.encode(payload)

if let userAgent = userAgent {
request.setValue(userAgent, forHTTPHeaderField: "User-Agent")
}

let session = URLSession(configuration: .ephemeral)
return await withCheckedContinuation { continuation in
let task = session.dataTask(with: request) { data, response, error in
var success: Bool = true
if let error = error {
Logger.module.error("Failed to report webcompat issue: \(error.localizedDescription)")
success = false
}
if let response = response as? HTTPURLResponse {
success = response.statusCode >= 200 && response.statusCode < 300
if !success {
Logger.module.error("Failed to report webcompat issue: Status Code \(response.statusCode)")
}
}
continuation.resume(returning: success)
let result = try await session.data(for: request)

if let response = result.1 as? HTTPURLResponse {
let success = response.statusCode >= 200 && response.statusCode < 300

if !success {
log.error("Failed to report webcompat issue: Status Code \(response.statusCode)")
}
task.resume()

return success
} else {
return false
}
} catch {
Logger.module.error("Failed to setup webcompat request payload: \(error.localizedDescription)")
return false
}
}
}

private extension ShieldLevel {
/// The value that is sent to the webcompat report server
var reportLabel: String {
switch self {
case .aggressive: return "aggressive"
case .standard: return "standard"
case .disabled: return "allow"
}
}
}
15 changes: 13 additions & 2 deletions Sources/Data/models/Domain.swift
Original file line number Diff line number Diff line change
Expand Up @@ -41,10 +41,12 @@ public final class Domain: NSManagedObject, CRUD {
/// A list of etld+1s that are always aggressive
private let alwaysAggressiveETLDs: Set<String> = ["youtube.com"]

/// Return the shield level for this domain
/// Return the shield level for this domain.
///
/// This only takes into consideration certain domains that are always aggressive.
/// - Warning: This does not consider the "all off" setting
/// This also takes into consideration certain domains that are always aggressive.
@MainActor public var blockAdsAndTrackingLevel: ShieldLevel {
guard isShieldExpected(.AdblockAndTp, considerAllShieldsOption: false) else { return .disabled }
let globalLevel = ShieldPreferences.blockAdsAndTrackingLevel

switch globalLevel {
Expand All @@ -63,6 +65,15 @@ public final class Domain: NSManagedObject, CRUD {
}
}

/// Return the finterprinting protection level for this domain.
///
/// - Warning: This does not consider the "all off" setting
@MainActor public var finterprintProtectionLevel: ShieldLevel {
guard isShieldExpected(.FpProtection, considerAllShieldsOption: false) else { return .disabled }
// We don't have aggressive finterprint protection in iOS
return .standard
}

private static let containsEthereumPermissionsPredicate = NSPredicate(format: "wallet_permittedAccounts != nil && wallet_permittedAccounts != ''")
private static let containsSolanaPermissionsPredicate = NSPredicate(format: "wallet_solanaPermittedAcccounts != nil && wallet_solanaPermittedAcccounts != ''")

Expand Down
4 changes: 2 additions & 2 deletions Tests/BraveSharedTests/URLExtensionTests.swift
Original file line number Diff line number Diff line change
Expand Up @@ -22,8 +22,8 @@ class URLExtensionTests: XCTestCase {
"http://test:t/est",
]

urls.forEach { XCTAssertEqual(URL(string: $0.0)!.origin.serialized, $0.1) }
badurls.forEach { XCTAssertTrue(URL(string: $0)!.origin.isOpaque) }
urls.forEach { XCTAssertEqual(URL(string: $0.0)?.origin.serialized, $0.1) }
badurls.forEach { XCTAssertTrue(URL(string: $0)?.origin.isOpaque ?? true) }
}

func testStrippedInternalURL() {
Expand Down

0 comments on commit a3aea32

Please sign in to comment.