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

Commit

Permalink
Fix #8100: Use proper ad-block components
Browse files Browse the repository at this point in the history
  • Loading branch information
cuba committed Oct 17, 2023
1 parent 35319a9 commit cd57c3d
Show file tree
Hide file tree
Showing 9 changed files with 387 additions and 90 deletions.
5 changes: 5 additions & 0 deletions Sources/Brave/WebFilters/FilterList.swift
Original file line number Diff line number Diff line change
Expand Up @@ -46,6 +46,11 @@ struct FilterList: Identifiable {
let entry: AdblockFilterListCatalogEntry
var isEnabled: Bool = false

/// Tells us if this filter list is regional (i.e. if it contains language restrictions)
var isRegional: Bool {
return !entry.languages.isEmpty
}

/// Lets us know if this filter list is always aggressive.
/// Aggressive filter lists are those that are non regional.
var isAlwaysAggressive: Bool { !isRegional }
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -53,7 +53,7 @@ actor FilterListCustomURLDownloader: ObservableObject {
}

/// Load any custom filter lists from cache so they are ready to use and start fetching updates.
func start() async {
func startIfNeeded() async {
guard !startedService else { return }
self.startedService = true
await CustomFilterListStorage.shared.loadCachedFilterLists()
Expand Down
4 changes: 0 additions & 4 deletions Sources/Brave/WebFilters/FilterListInterface.swift
Original file line number Diff line number Diff line change
Expand Up @@ -23,8 +23,4 @@ extension FilterList: FilterListInterface {
var debugTitle: String {
return "\(entry.title) \(entry.componentId)"
}

var isRegional: Bool {
return !entry.languages.isEmpty
}
}
186 changes: 112 additions & 74 deletions Sources/Brave/WebFilters/FilterListResourceDownloader.swift
Original file line number Diff line number Diff line change
Expand Up @@ -25,7 +25,7 @@ public actor FilterListResourceDownloader {
private var adBlockServiceTasks: [String: Task<Void, Error>]
/// A marker that says that we loaded shield components for the first time.
/// This boolean is used to configure this downloader only once after `AdBlockService` generic shields have been loaded.
private var loadedShieldComponents = false
private var registeredFilterLists = false
/// The path to the resources file
private(set) var resourcesInfo: CachedAdBlockEngine.ResourcesInfo?

Expand All @@ -40,23 +40,27 @@ public actor FilterListResourceDownloader {
/// - Warning: This method loads filter list settings.
/// You need to wait for `DataController.shared.initializeOnce()` to be called first before invoking this method
public func loadFilterListSettingsAndCachedData() async {
guard let folderURL = await FilterListSetting.makeFolderURL(
forFilterListFolderPath: Preferences.AppState.lastDefaultFilterListFolderPath.value
), FileManager.default.fileExists(atPath: folderURL.path) else {
return
if let defaultFilterListFolderURL = await FilterListSetting.makeFolderURL(
forFilterListFolderPath: Preferences.AppState.lastFilterListCatalogueComponentFolderPath.value
), FileManager.default.fileExists(atPath: defaultFilterListFolderURL.path),
let resourcesFolderURL = await FilterListSetting.makeFolderURL(
forFilterListFolderPath: Preferences.AppState.lastAdBlockResourcesFolderPath.value
), FileManager.default.fileExists(atPath: resourcesFolderURL.path) {
let resourcesInfo = await didUpdateResourcesComponent(folderURL: resourcesFolderURL)
async let startedCustomFilterListsDownloader: Void = FilterListCustomURLDownloader.shared.startIfNeeded()
async let cachedFilterLists: Void = compileCachedFilterLists(resourcesInfo: resourcesInfo)
async let compileDefaultEngine: Void = compileDefaultEngine(defaultFilterListFolderURL: defaultFilterListFolderURL, resourcesInfo: resourcesInfo)
_ = await (startedCustomFilterListsDownloader, cachedFilterLists, compileDefaultEngine)
} else if let legacyComponentFolderURL = await FilterListSetting.makeFolderURL(
forFilterListFolderPath: Preferences.AppState.lastLegacyDefaultFilterListFolderPath.value
), FileManager.default.fileExists(atPath: legacyComponentFolderURL.path) {
// TODO: @JS Remove this after this release. Its here just so users can upgrade without a pause to their adblocking
let resourcesInfo = await didUpdateResourcesComponent(folderURL: legacyComponentFolderURL)
async let startedCustomFilterListsDownloader: Void = FilterListCustomURLDownloader.shared.startIfNeeded()
async let cachedFilterLists: Void = compileCachedFilterLists(resourcesInfo: resourcesInfo)
async let compileDefaultEngine: Void = compileDefaultEngine(defaultFilterListFolderURL: legacyComponentFolderURL, resourcesInfo: resourcesInfo)
_ = await (startedCustomFilterListsDownloader, cachedFilterLists, compileDefaultEngine)
}

let version = folderURL.lastPathComponent
let resourcesInfo = CachedAdBlockEngine.ResourcesInfo(
localFileURL: folderURL.appendingPathComponent("resources.json", conformingTo: .json),
version: version
)
self.resourcesInfo = resourcesInfo

async let startedCustomFilterListsDownloader: Void = FilterListCustomURLDownloader.shared.start()
async let cachedFilterLists: Void = compileCachedFilterLists(resourcesInfo: resourcesInfo)
async let compileDefaultEngine: Void = compileDefaultEngine(shieldsInstallFolder: folderURL, resourcesInfo: resourcesInfo)
_ = await (startedCustomFilterListsDownloader, cachedFilterLists, compileDefaultEngine)
}

/// This function adds engine resources to `AdBlockManager` from cached data representing the enabled filter lists.
Expand Down Expand Up @@ -103,59 +107,94 @@ public actor FilterListResourceDownloader {

// Start listening to changes to the install url
Task { @MainActor in
for await folderURL in adBlockService.shieldsInstallURL {
await self.didUpdateShieldComponent(
folderURL: folderURL,
adBlockFilterLists: adBlockService.regionalFilterLists ?? []
)
for await folderURL in adBlockService.resourcesComponentStream() {
guard let folderURL = folderURL else {
ContentBlockerManager.log.error("Missing folder for filter lists")
return
}

await didUpdateResourcesComponent(folderURL: folderURL)
await FilterListCustomURLDownloader.shared.startIfNeeded()

if !FilterListStorage.shared.filterLists.isEmpty {
await registerAllFilterListsIfNeeded(with: adBlockService)
}
}
}

Task { @MainActor in
for await filterListEntries in adBlockService.filterListCatalogComponentStream() {
FilterListStorage.shared.loadFilterLists(from: filterListEntries)

if await self.resourcesInfo != nil {
await registerAllFilterListsIfNeeded(with: adBlockService)
}
}
}
}

/// Invoked when shield components are loaded
///
/// This function will start fetching data and subscribe publishers once if it hasn't already done so.
private func didUpdateShieldComponent(folderURL: URL, adBlockFilterLists: [AdblockFilterListCatalogEntry]) async {
// Store the folder path so we can load it from cache next time we launch quicker
// than waiting for the component updater to respond, which may take a few seconds
/// Register all enabled filter lists and to the default filter list with the `AdBlockService`
private func registerAllFilterListsIfNeeded(with adBlockService: AdblockService) async {
guard !registeredFilterLists else { return }
self.registeredFilterLists = true
registerToDefaultFilterList(with: adBlockService)

for filterList in await FilterListStorage.shared.filterLists {
register(filterList: filterList)
}
}

/// Register to changes to the default filter list with the given ad-block service
private func registerToDefaultFilterList(with adBlockService: AdblockService) {
// Register the default filter list
Task { @MainActor in
for await folderURL in adBlockService.defaultComponentStream() {
guard let folderURL = folderURL else {
ContentBlockerManager.log.error("Missing folder for filter lists")
return
}

await Task { @MainActor in
let folderSubPath = FilterListSetting.extractFolderPath(fromFilterListFolderURL: folderURL)
Preferences.AppState.lastFilterListCatalogueComponentFolderPath.value = folderSubPath
}.value

if let resourcesInfo = await self.resourcesInfo {
await compileDefaultEngine(defaultFilterListFolderURL: folderURL, resourcesInfo: resourcesInfo)
}
}
}
}

@discardableResult
/// When the
private func didUpdateResourcesComponent(folderURL: URL) async -> CachedAdBlockEngine.ResourcesInfo {
await Task { @MainActor in
let folderSubPath = FilterListSetting.extractFolderPath(fromFilterListFolderURL: folderURL)
Preferences.AppState.lastDefaultFilterListFolderPath.value = folderSubPath
Preferences.AppState.lastAdBlockResourcesFolderPath.value = folderSubPath
}.value

// Set the resources info so other filter lists can use them
let version = folderURL.lastPathComponent
let resourcesInfo = CachedAdBlockEngine.ResourcesInfo(
localFileURL: folderURL.appendingPathComponent("resources.json", conformingTo: .json),
version: version
)
self.resourcesInfo = resourcesInfo

// Perform one time setup
if !loadedShieldComponents && !adBlockFilterLists.isEmpty {
// This is the first time we load ad-block filters.
// We need to perform some initial setup (but only do this once)
loadedShieldComponents = true
await FilterListStorage.shared.loadFilterLists(from: adBlockFilterLists)

Task {
// Start the custom filter list downloader
await FilterListCustomURLDownloader.shared.start()
}
}

// Compile the engine
await compileDefaultEngine(shieldsInstallFolder: folderURL, resourcesInfo: resourcesInfo)
await registerAllFilterLists()
self.resourcesInfo = resourcesInfo
return resourcesInfo
}

/// Compile the general engine from the given `AdblockService` `shieldsInstallPath` `URL`.
private func compileDefaultEngine(shieldsInstallFolder folderURL: URL, resourcesInfo: CachedAdBlockEngine.ResourcesInfo) async {
private func compileDefaultEngine(defaultFilterListFolderURL folderURL: URL, resourcesInfo: CachedAdBlockEngine.ResourcesInfo) async {
// TODO: @JS Remove this on the next update. This is here so users don't have a pause to their ad-blocking
let isLegacy = folderURL.pathExtension == "dat"
let localFileURL = isLegacy ? folderURL.appendingPathComponent("rs-ABPFilterParserData.dat", conformingTo: .data) : folderURL.appendingPathComponent("list.txt", conformingTo: .text)

let version = folderURL.lastPathComponent
let filterListInfo = CachedAdBlockEngine.FilterListInfo(
source: .adBlock,
localFileURL: folderURL.appendingPathComponent("rs-ABPFilterParserData.dat", conformingTo: .data),
version: version, fileType: .dat
localFileURL: localFileURL,
version: version, fileType: isLegacy ? .dat : .text
)

guard await AdBlockStats.shared.needsCompilation(for: filterListInfo, resourcesInfo: resourcesInfo) else {
Expand Down Expand Up @@ -206,13 +245,6 @@ public actor FilterListResourceDownloader {
}
}

/// Register all enabled filter lists with the `AdBlockService`
@MainActor private func registerAllFilterLists() async {
for filterList in FilterListStorage.shared.filterLists {
await register(filterList: filterList)
}
}

/// Register this filter list with the `AdBlockService`
private func register(filterList: FilterList) {
guard adBlockServiceTasks[filterList.entry.componentId] == nil else { return }
Expand Down Expand Up @@ -307,32 +339,38 @@ public actor FilterListResourceDownloader {

/// Helpful extension to the AdblockService
private extension AdblockService {
/// Stream the URL updates to the `shieldsInstallPath`
///
/// - Warning: You should never do this more than once. Only one callback can be registered to the `shieldsComponentReady` callback.
@MainActor var shieldsInstallURL: AsyncStream<URL> {
@MainActor func defaultComponentStream() -> AsyncStream<URL?> {
return AsyncStream { continuation in
if let folderPath = shieldsInstallPath {
registerDefaultComponent { folderPath in
guard let folderPath = folderPath else {
continuation.yield(nil)
return
}

let folderURL = URL(fileURLWithPath: folderPath)
continuation.yield(folderURL)
}

guard shieldsComponentReady == nil else {
assertionFailure("You have already set the `shieldsComponentReady` callback. Setting this more than once replaces the previous callback.")
return
}

shieldsComponentReady = { folderPath in
}
}

@MainActor func resourcesComponentStream() -> AsyncStream<URL?> {
return AsyncStream { continuation in
registerResourceComponent { folderPath in
guard let folderPath = folderPath else {
continuation.yield(nil)
return
}

let folderURL = URL(fileURLWithPath: folderPath)
continuation.yield(folderURL)
}

continuation.onTermination = { @Sendable _ in
self.shieldsComponentReady = nil
}
}

@MainActor func filterListCatalogComponentStream() -> AsyncStream<[AdblockFilterListCatalogEntry]> {
return AsyncStream { continuation in
registerFilterListCatalogComponent { filterListEntries in
continuation.yield(filterListEntries)
}
}
}
Expand All @@ -342,7 +380,7 @@ private extension AdblockService {
/// - Note: Cancelling this task will unregister this filter list from recieving any further updates
@MainActor func register(filterList: FilterList) -> AsyncStream<URL?> {
return AsyncStream { continuation in
registerFilterListComponent(filterList.entry, useLegacyComponent: false) { folderPath in
registerFilterListComponent(filterList.entry) { folderPath in
guard let folderPath = folderPath else {
continuation.yield(nil)
return
Expand All @@ -353,7 +391,7 @@ private extension AdblockService {
}

continuation.onTermination = { @Sendable _ in
self.unregisterFilterListComponent(filterList.entry, useLegacyComponent: true)
self.unregisterFilterListComponent(filterList.entry)
}
}
}
Expand Down
24 changes: 16 additions & 8 deletions Sources/Brave/WebFilters/FilterListStorage.swift
Original file line number Diff line number Diff line change
Expand Up @@ -145,6 +145,7 @@ import Combine
upsertSetting(
uuid: filterList.entry.uuid,
isEnabled: filterList.isEnabled,
isHidden: false,
componentId: filterList.entry.componentId,
allowCreation: true,
order: filterList.order,
Expand All @@ -157,14 +158,15 @@ import Combine
///
/// - Warning: Do not call this before we load core data
private func upsertSetting(
uuid: String, isEnabled: Bool, componentId: String,
uuid: String, isEnabled: Bool, isHidden: Bool, componentId: String,
allowCreation: Bool, order: Int, isAlwaysAggressive: Bool
) {
if allFilterListSettings.contains(where: { $0.uuid == uuid }) {
updateSetting(
uuid: uuid,
componentId: componentId,
isEnabled: isEnabled,
isHidden: isHidden,
order: order,
isAlwaysAggressive: isAlwaysAggressive
)
Expand All @@ -173,6 +175,7 @@ import Combine
uuid: uuid,
componentId: componentId,
isEnabled: isEnabled,
isHidden: isHidden,
order: order,
isAlwaysAggressive: isAlwaysAggressive
)
Expand All @@ -194,29 +197,34 @@ import Combine

/// Update the filter list settings with the given `componentId` and `isEnabled` status
/// Will not write unless one of these two values have changed
private func updateSetting(uuid: String, componentId: String, isEnabled: Bool, order: Int, isAlwaysAggressive: Bool) {
private func updateSetting(uuid: String, componentId: String, isEnabled: Bool, isHidden: Bool, order: Int, isAlwaysAggressive: Bool) {
guard let index = allFilterListSettings.firstIndex(where: { $0.uuid == uuid }) else {
return
}

guard allFilterListSettings[index].isEnabled != isEnabled || allFilterListSettings[index].componentId != componentId || allFilterListSettings[index].order?.intValue != order || allFilterListSettings[index].isAlwaysAggressive != isAlwaysAggressive else {
// Ensure we stop if this is already in sync in order to avoid an event loop
// And things hanging for too long.
// This happens because we care about UI changes but not when our downloads finish
// Ensure we stop if this is already in sync in order to avoid an event loop
// And things hanging for too long.
guard allFilterListSettings[index].isEnabled != isEnabled
|| allFilterListSettings[index].componentId != componentId
|| allFilterListSettings[index].order?.intValue != order
|| allFilterListSettings[index].isAlwaysAggressive != isAlwaysAggressive
|| allFilterListSettings[index].isHidden != isHidden
else {
return
}

allFilterListSettings[index].isEnabled = isEnabled
allFilterListSettings[index].isAlwaysAggressive = isAlwaysAggressive
allFilterListSettings[index].isHidden = isHidden
allFilterListSettings[index].componentId = componentId
allFilterListSettings[index].order = NSNumber(value: order)
FilterListSetting.save(inMemory: !persistChanges)
}

/// Create a filter list setting for the given UUID and enabled status
private func create(uuid: String, componentId: String, isEnabled: Bool, order: Int, isAlwaysAggressive: Bool) {
private func create(uuid: String, componentId: String, isEnabled: Bool, isHidden: Bool, order: Int, isAlwaysAggressive: Bool) {
let setting = FilterListSetting.create(
uuid: uuid, componentId: componentId, isEnabled: isEnabled, order: order, inMemory: !persistChanges,
uuid: uuid, componentId: componentId, isEnabled: isEnabled, isHidden: isHidden, order: order, inMemory: !persistChanges,
isAlwaysAggressive: isAlwaysAggressive
)
allFilterListSettings.append(setting)
Expand Down
4 changes: 3 additions & 1 deletion Sources/Data/models/FilterListSetting.swift
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@ public final class FilterListSetting: NSManagedObject, CRUD {
@MainActor @NSManaged public var uuid: String
@MainActor @NSManaged public var componentId: String?
@MainActor @NSManaged public var isEnabled: Bool
@MainActor @NSManaged public var isHidden: Bool
@MainActor @NSManaged public var isAlwaysAggressive: Bool
@MainActor @NSManaged public var order: NSNumber?
@MainActor @NSManaged private var folderPath: String?
Expand All @@ -39,7 +40,7 @@ public final class FilterListSetting: NSManagedObject, CRUD {

/// Create a filter list setting for the given UUID and enabled status
@MainActor public class func create(
uuid: String, componentId: String?, isEnabled: Bool, order: Int, inMemory: Bool, isAlwaysAggressive: Bool
uuid: String, componentId: String?, isEnabled: Bool, isHidden: Bool, order: Int, inMemory: Bool, isAlwaysAggressive: Bool
) -> FilterListSetting {
var newSetting: FilterListSetting!

Expand All @@ -52,6 +53,7 @@ public final class FilterListSetting: NSManagedObject, CRUD {
newSetting.uuid = uuid
newSetting.componentId = componentId
newSetting.isEnabled = isEnabled
newSetting.isHidden = isHidden
newSetting.isAlwaysAggressive = isAlwaysAggressive
newSetting.order = NSNumber(value: order)
}
Expand Down
Loading

0 comments on commit cd57c3d

Please sign in to comment.