Skip to content
This repository has been archived by the owner on May 10, 2024. It is now read-only.

Fix compiling only necessary files during launch #8749

Closed
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -149,7 +149,6 @@ extension BrowserViewController: WKNavigationDelegate {

@MainActor
public func webView(_ webView: WKWebView, decidePolicyFor navigationAction: WKNavigationAction, preferences: WKWebpagePreferences) async -> (WKNavigationActionPolicy, WKWebpagePreferences) {

guard var requestURL = navigationAction.request.url else {
return (.cancel, preferences)
}
Expand Down Expand Up @@ -255,6 +254,9 @@ extension BrowserViewController: WKNavigationDelegate {
return (.cancel, preferences)
}

let signpostID = ContentBlockerManager.signpost.makeSignpostID()
let state = ContentBlockerManager.signpost.beginInterval("decidePolicyFor", id: signpostID)

// before loading any ad-block scripts
// await the preparation of the ad-block services
await LaunchHelper.shared.prepareAdBlockServices(
Expand All @@ -279,6 +281,7 @@ extension BrowserViewController: WKNavigationDelegate {
ContentBlockerManager.log.debug("Redirected user to `\(url.absoluteString, privacy: .private)`")
}

ContentBlockerManager.signpost.endInterval("decidePolicyFor", state, "Redirected navigation")
return (.cancel, preferences)
} else {
tab?.isInternalRedirect = false
Expand Down Expand Up @@ -321,6 +324,7 @@ extension BrowserViewController: WKNavigationDelegate {
var modifiedRequest = URLRequest(url: requestURL)
modifiedRequest.setValue("1", forHTTPHeaderField: "X-Brave-Ads-Enabled")
tab?.loadRequest(modifiedRequest)
ContentBlockerManager.signpost.endInterval("decidePolicyFor", state, "Redirected to search")
return (.cancel, preferences)
}

Expand Down Expand Up @@ -373,6 +377,7 @@ extension BrowserViewController: WKNavigationDelegate {
if let url = components?.url {
let request = PrivilegedRequest(url: url) as URLRequest
tab?.loadRequest(request)
ContentBlockerManager.signpost.endInterval("decidePolicyFor", state, "Blocked navigation")
return (.cancel, preferences)
}
}
Expand Down Expand Up @@ -437,6 +442,7 @@ extension BrowserViewController: WKNavigationDelegate {
self.shouldDownloadNavigationResponse = true
}

ContentBlockerManager.signpost.endInterval("decidePolicyFor", state)
return (.allow, preferences)
}

Expand Down
4 changes: 3 additions & 1 deletion Sources/Brave/Frontend/Browser/Helpers/LaunchHelper.swift
Original file line number Diff line number Diff line change
Expand Up @@ -39,6 +39,7 @@ public actor LaunchHelper {
// Otherwise prepare the services and await the task
let task = Task {
let signpostID = Self.signpost.makeSignpostID()
ContentBlockerManager.log.debug("Loading blocking launch data")
let state = Self.signpost.beginInterval("blockingLaunchTask", id: signpostID)
// We only want to compile the necessary content blockers during launch
// We will compile other ones after launch
Expand All @@ -50,6 +51,7 @@ public actor LaunchHelper {
async let adblockResourceCache: Void = AdblockResourceDownloader.shared.loadCachedAndBundledDataIfNeeded(allowedModes: launchBlockModes)
_ = await (filterListCache, adblockResourceCache)
Self.signpost.emitEvent("loadedCachedData", id: signpostID, "Loaded cached data")
ContentBlockerManager.log.debug("Loaded blocking launch data")

// This one is non-blocking
performPostLoadTasks(adBlockService: adBlockService, loadedBlockModes: launchBlockModes)
Expand Down Expand Up @@ -128,7 +130,7 @@ private extension FilterListStorage {
var validBlocklistTypes: Set<ContentBlockerManager.BlocklistType> {
if filterLists.isEmpty {
// If we don't have filter lists yet loaded, use the settings
return Set(allFilterListSettings.compactMap { setting in
return Set(allFilterListSettings.compactMap { setting -> ContentBlockerManager.BlocklistType? in
guard let componentId = setting.componentId else { return nil }
return .filterList(
componentId: componentId,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -92,7 +92,7 @@ extension UserScriptType: CustomDebugStringConvertible {
case .nacl:
return "nacl"
case .gpc(let isEnabled):
return "gpc(\(isEnabled)"
return "gpc(\(isEnabled))"
case .siteStateListener:
return "siteStateListener"
case .selectorsPoller:
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -153,6 +153,8 @@ actor ContentBlockerManager {
/// Should be used only as a cleanup once during launch to get rid of unecessary/old data.
/// This is mostly for custom filter lists a user may have added.
public func cleaupInvalidRuleLists(validTypes: Set<BlocklistType>) async {
let signpostID = Self.signpost.makeSignpostID()
let state = Self.signpost.beginInterval("cleaupInvalidRuleLists", id: signpostID)
let availableIdentifiers = await ruleStore.availableIdentifiers() ?? []

await availableIdentifiers.asyncConcurrentForEach { identifier in
Expand All @@ -170,6 +172,8 @@ actor ContentBlockerManager {
assertionFailure()
}
}

Self.signpost.endInterval("cleaupInvalidRuleLists", state)
}

/// Compile the rule list found in the given local URL using the specified modes
Expand Down
4 changes: 4 additions & 0 deletions Sources/Brave/WebFilters/FilterList.swift
Original file line number Diff line number Diff line change
Expand Up @@ -46,6 +46,10 @@ struct FilterList: Identifiable {
let entry: AdblockFilterListCatalogEntry
var isEnabled: Bool = false

var isHidden: Bool {
// TODO: @JS get this from entry.hidden once it is available.
return false
}
/// Tells us if this filter list is regional (i.e. if it contains language restrictions)
var isRegional: Bool {
return !entry.languages.isEmpty
Expand Down
29 changes: 4 additions & 25 deletions Sources/Brave/WebFilters/FilterListCustomURLDownloader.swift
Original file line number Diff line number Diff line change
Expand Up @@ -58,29 +58,9 @@ actor FilterListCustomURLDownloader: ObservableObject {
self.startedService = true
await CustomFilterListStorage.shared.loadCachedFilterLists()

do {
try await CustomFilterListStorage.shared.filterListsURLs.asyncConcurrentForEach { customURL in
let resource = await customURL.setting.resource

do {
if let cachedResult = try resource.cachedResult() {
await self.handle(downloadResult: cachedResult, for: customURL)
}
} catch {
let uuid = await customURL.setting.uuid
ContentBlockerManager.log.error(
"Failed to cached data for custom filter list `\(uuid)`: \(error)"
)
}

// Always fetch this resource so it's ready if the user enables it.
await self.startFetching(filterListCustomURL: customURL)

// Sleep for 1ms. This drastically reduces memory usage without much impact to usability
try await Task.sleep(nanoseconds: 1000000)
}
} catch {
// Ignore cancellation errors
await CustomFilterListStorage.shared.filterListsURLs.asyncForEach { customURL in
// Always fetch this resource so it's ready if the user enables it.
await self.startFetching(filterListCustomURL: customURL)
}
}

Expand All @@ -105,9 +85,8 @@ actor FilterListCustomURLDownloader: ObservableObject {
filterListInfo: filterListInfo, isAlwaysAggressive: true
)

guard await AdBlockStats.shared.isEagerlyLoaded(source: source) else {
guard await AdBlockStats.shared.isEnabled(source: source) else {
// Don't compile unless eager
await AdBlockStats.shared.updateIfNeeded(resourcesInfo: resourcesInfo)
await AdBlockStats.shared.updateIfNeeded(filterListInfo: filterListInfo, isAlwaysAggressive: true)

// To free some space, remove any rule lists that are not needed
Expand Down
73 changes: 26 additions & 47 deletions Sources/Brave/WebFilters/FilterListResourceDownloader.swift
Original file line number Diff line number Diff line change
Expand Up @@ -46,11 +46,9 @@ public actor FilterListResourceDownloader {
}

let resourcesInfo = await didUpdateResourcesComponent(folderURL: resourcesFolderURL)
async let startedCustomFilterListsDownloader: Void = FilterListCustomURLDownloader.shared.startIfNeeded()
async let cachedFilterLists: Void = compileCachedFilterLists(resourcesInfo: resourcesInfo)
async let compileDefaultEngine: Void = compileDefaultFilterList(resourcesInfo: resourcesInfo)

_ = await (startedCustomFilterListsDownloader, cachedFilterLists, compileDefaultEngine)
_ = await (cachedFilterLists, compileDefaultEngine)
}

/// Compile the default filter list from cache
Expand Down Expand Up @@ -86,17 +84,18 @@ public actor FilterListResourceDownloader {

do {
try await filterListSettings.asyncConcurrentForEach { setting in
guard await setting.isEnabled == true else { return }
guard await setting.isEagerlyLoaded == true else { return }
guard let componentId = await setting.componentId else { return }
guard FilterList.disabledComponentIDs.contains(componentId) else { return }
guard !FilterList.disabledComponentIDs.contains(componentId) else { return }
guard let source = await setting.engineSource else { return }

// Try to load the filter list folder. We always have to compile this at start
guard let folderURL = await setting.folderURL, FileManager.default.fileExists(atPath: folderURL.path) else {
return
}

await self.compileFilterListEngineIfNeeded(
fromComponentId: componentId, folderURL: folderURL,
source: source, folderURL: folderURL,
isAlwaysAggressive: setting.isAlwaysAggressive,
resourcesInfo: resourcesInfo,
compileContentBlockers: false
Expand Down Expand Up @@ -138,7 +137,7 @@ public actor FilterListResourceDownloader {
Task { @MainActor in
for await filterListEntries in adBlockService.filterListCatalogComponentStream() {
FilterListStorage.shared.loadFilterLists(from: filterListEntries)

ContentBlockerManager.log.debug("Loaded filter list catalog")
if await AdBlockStats.shared.resourcesInfo != nil {
await registerAllFilterListsIfNeeded(with: adBlockService)
}
Expand Down Expand Up @@ -227,22 +226,30 @@ public actor FilterListResourceDownloader {

/// Load general filter lists (shields) from the given `AdblockService` `shieldsInstallPath` `URL`.
private func compileFilterListEngineIfNeeded(
fromComponentId componentId: String, folderURL: URL,
source: CachedAdBlockEngine.Source,
folderURL: URL,
isAlwaysAggressive: Bool,
resourcesInfo: CachedAdBlockEngine.ResourcesInfo,
compileContentBlockers: Bool
) async {
let version = folderURL.lastPathComponent
let source = CachedAdBlockEngine.Source.filterList(componentId: componentId)
let filterListURL = folderURL.appendingPathComponent("list.txt", conformingTo: .text)

guard FileManager.default.fileExists(atPath: filterListURL.relativePath) else {
// We are loading the old component from cache. We don't want this file to be loaded.
// When we download the new component shortly we will update our cache.
// This should only trigger after an app update and eventually this check can be removed.
return
}

let filterListInfo = CachedAdBlockEngine.FilterListInfo(
source: source,
localFileURL: folderURL.appendingPathComponent("list.txt", conformingTo: .text),
source: source, localFileURL: filterListURL,
version: version, fileType: .text
)
let lazyInfo = AdBlockStats.LazyFilterListInfo(filterListInfo: filterListInfo, isAlwaysAggressive: isAlwaysAggressive)
guard await AdBlockStats.shared.isEagerlyLoaded(source: source) else {
// Don't compile unless eager
await AdBlockStats.shared.updateIfNeeded(resourcesInfo: resourcesInfo)

// Check if we should load these rules
guard await AdBlockStats.shared.isEnabled(source: source) else {
await AdBlockStats.shared.updateIfNeeded(filterListInfo: filterListInfo, isAlwaysAggressive: isAlwaysAggressive)

// To free some space, remove any rule lists that are not needed
Expand Down Expand Up @@ -275,47 +282,19 @@ public actor FilterListResourceDownloader {
assertionFailure("We shouldn't have started downloads before getting this value")
return
}
let source = filterList.engineSource

await self.loadShields(
fromComponentId: filterList.entry.componentId, folderURL: folderURL, relativeOrder: filterList.order,
loadContentBlockers: true,
isAlwaysAggressive: filterList.isAlwaysAggressive,
resourcesInfo: resourcesInfo
// Add or remove the filter list from the engine depending if it's been enabled or not
await self.compileFilterListEngineIfNeeded(
source: source, folderURL: folderURL, isAlwaysAggressive: filterList.isAlwaysAggressive,
resourcesInfo: resourcesInfo, compileContentBlockers: true
)

// Save the downloaded folder for later (caching) purposes
FilterListStorage.shared.set(folderURL: folderURL, forUUID: filterList.entry.uuid)
}
}
}

/// Handle the downloaded component folder url of a filter list.
///
/// The folder URL should point to a `AdblockFilterListEntry` download location as given by the `AdBlockService`.
///
/// If `loadContentBlockers` is set to `true`, this method will compile the rule lists to content blocker format and load them into the `WKContentRuleListStore`.
/// As both these procedures are expensive, this should be set to `false` if this method is called on a blocking UI process such as the launch of the application.
private func loadShields(
fromComponentId componentId: String, folderURL: URL, relativeOrder: Int, loadContentBlockers: Bool, isAlwaysAggressive: Bool,
resourcesInfo: CachedAdBlockEngine.ResourcesInfo
) async {
// Check if we're loading the new component or an old component from cache.
// The new component has a file `list.txt` which we check the presence of.
let filterListURL = folderURL.appendingPathComponent("list.txt", conformingTo: .text)

guard FileManager.default.fileExists(atPath: filterListURL.relativePath) else {
// We are loading the old component from cache. We don't want this file to be loaded.
// When we download the new component shortly we will update our cache.
// This should only trigger after an app update and eventually this check can be removed.
return
}

// Add or remove the filter list from the engine depending if it's been enabled or not
await self.compileFilterListEngineIfNeeded(
fromComponentId: componentId, folderURL: folderURL, isAlwaysAggressive: isAlwaysAggressive,
resourcesInfo: resourcesInfo, compileContentBlockers: loadContentBlockers
)
}
}

/// Helpful extension to the AdblockService
Expand Down
2 changes: 1 addition & 1 deletion Sources/Brave/WebFilters/FilterListStorage.swift
Original file line number Diff line number Diff line change
Expand Up @@ -134,7 +134,7 @@ import Combine
upsertSetting(
uuid: filterList.entry.uuid,
isEnabled: filterList.isEnabled,
isHidden: false,
isHidden: filterList.isHidden,
componentId: filterList.entry.componentId,
allowCreation: true,
order: filterList.order,
Expand Down
39 changes: 8 additions & 31 deletions Sources/Brave/WebFilters/ShieldStats/Adblock/AdBlockStats.swift
Original file line number Diff line number Diff line change
Expand Up @@ -29,16 +29,6 @@ public actor AdBlockStats {
/// The current task that is compiling.
private var currentCompileTask: Task<(), Never>?

/// Return all the critical sources
///
/// Critical sources are those that are enabled and are "on" by default. Giving us the most important sources.
/// Used for memory managment so we know which filter lists to disable upon a memory warning
@MainActor var criticalSources: [CachedAdBlockEngine.Source] {
var enabledSources: [CachedAdBlockEngine.Source] = [.adBlock]
enabledSources.append(contentsOf: FilterListStorage.shared.criticalSources)
return enabledSources
}

/// Return an array of all sources that are enabled according to user's settings
/// - Note: This does not take into account the domain or global adblock toggle
@MainActor var enabledSources: [CachedAdBlockEngine.Source] {
Expand Down Expand Up @@ -125,6 +115,12 @@ public actor AdBlockStats {
func updateIfNeeded(resourcesInfo: CachedAdBlockEngine.ResourcesInfo) {
guard self.resourcesInfo == nil || resourcesInfo.version > self.resourcesInfo!.version else { return }
self.resourcesInfo = resourcesInfo

if #available(iOS 16.0, *) {
ContentBlockerManager.log.debug(
"Updated resources component: `\(resourcesInfo.localFileURL.path(percentEncoded: false))`"
)
}
}

/// Remove all the engines
Expand Down Expand Up @@ -177,12 +173,8 @@ public actor AdBlockStats {
}
}

/// Tells us if this source should be eagerly loaded.
///
/// Eagerness is determined by several factors:
/// * If the source represents a fitler list or a custom filter list, it is eager if it is enabled
/// * If the source represents the `adblock` default filter list, it is always eager regardless of shield settings
@MainActor func isEagerlyLoaded(source: CachedAdBlockEngine.Source) -> Bool {
/// Tells us if this source should be loaded.
@MainActor func isEnabled(source: CachedAdBlockEngine.Source) -> Bool {
return enabledSources.contains(source)
}

Expand Down Expand Up @@ -275,21 +267,6 @@ extension FilterList {
}

private extension FilterListStorage {
/// Gives us source representations of all the critical filter lists
///
/// Critical filter lists are those that are enabled and are "on" by default. Giving us the most important filter lists.
/// Used for memory managment so we know which filter lists to disable upon a memory warning
@MainActor var criticalSources: [CachedAdBlockEngine.Source] {
return enabledSources.filter { source in
switch source {
case .filterList(let componentId):
return FilterList.defaultOnComponentIds.contains(componentId)
default:
return false
}
}
}

/// Gives us source representations of all the enabled filter lists
@MainActor var enabledSources: [CachedAdBlockEngine.Source] {
if !filterLists.isEmpty {
Expand Down
Loading
Loading