Skip to content

Commit

Permalink
Closes mozilla-mobile#10245: Migrate tracking protection exceptions t…
Browse files Browse the repository at this point in the history
…o new GV storage
  • Loading branch information
Amejia481 authored and Grisha Kruglov committed Sep 11, 2021
1 parent 9fe5c51 commit 7e25272
Show file tree
Hide file tree
Showing 4 changed files with 257 additions and 228 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -14,15 +14,18 @@ import mozilla.components.browser.engine.gecko.content.blocking.GeckoTrackingPro
import mozilla.components.concept.engine.EngineSession
import mozilla.components.concept.engine.content.blocking.TrackingProtectionException
import mozilla.components.concept.engine.content.blocking.TrackingProtectionExceptionStorage
import mozilla.components.support.base.log.logger.Logger
import mozilla.components.support.ktx.util.readAndDeserialize
import mozilla.components.support.ktx.util.writeString
import org.json.JSONArray
import org.json.JSONObject
import org.mozilla.geckoview.ContentBlockingController.ContentBlockingException
import org.mozilla.geckoview.GeckoRuntime
import org.mozilla.geckoview.GeckoSession.PermissionDelegate.ContentPermission
import org.mozilla.geckoview.GeckoSession.PermissionDelegate.ContentPermission.VALUE_ALLOW
import org.mozilla.geckoview.GeckoSession.PermissionDelegate.ContentPermission.VALUE_DENY
import org.mozilla.geckoview.GeckoSession.PermissionDelegate.PERMISSION_TRACKING
import java.io.File

private const val STORE_FILE_NAME_FORMAT =
internal const val STORE_FILE_NAME =
"mozilla_components_tracking_protection_storage_gecko.json"

/**
Expand All @@ -34,119 +37,145 @@ internal class TrackingProtectionExceptionFileStorage(
) : TrackingProtectionExceptionStorage {
private val fileLock = Any()
internal var scope = CoroutineScope(Dispatchers.IO)
private val logger = Logger("TrackingProtectionExceptionFileStorage")

/**
* Restore all exceptions from the [STORE_FILE_NAME_FORMAT] file,
* Restore all exceptions from the [STORE_FILE_NAME] file,
* and provides them to the gecko [runtime].
*/
override fun restore() {
scope.launch {
synchronized(fileLock) {
getFile(context).readAndDeserialize { json ->
if (json.isNotEmpty()) {
val jsonArray = JSONArray(json)
val exceptionList = (0 until jsonArray.length()).map { index ->
val jsonObject = jsonArray.getJSONObject(index)
ContentBlockingException.fromJson(jsonObject)
}
runtime.contentBlockingController.restoreExceptionList(exceptionList)
}
}
}
if (!isMigrationOver(context)) {
logger.info("Starting tracking protection exceptions migration")
migrateExceptions()
}
}

override fun contains(session: EngineSession, onResult: (Boolean) -> Unit) {
val geckoSession = (session as GeckoEngineSession).geckoSession
runtime.contentBlockingController.checkException(geckoSession).accept {
if (it != null) {
onResult(it)
} else {
onResult(false)
val url = (session as GeckoEngineSession).currentUrl
if (!url.isNullOrEmpty()) {
runtime.storageController.getPermissions(url).accept { permissions ->
val contains = permissions.filterTrackingProtectionExceptions().isNotEmpty()
onResult(contains)
}
} else {
onResult(false)
}
}

override fun fetchAll(onResult: (List<TrackingProtectionException>) -> Unit) {
runtime.contentBlockingController.saveExceptionList().accept { exceptionList ->
val exceptions = if (exceptionList != null) {
val exceptions = exceptionList.map { it.toTrackingProtectionException() }
exceptions
} else {
emptyList()
}
onResult(exceptions)
runtime.storageController.allPermissions.accept { permissions ->
val trackingExceptions = permissions.filterTrackingProtectionExceptions()
onResult(trackingExceptions.map { exceptions -> exceptions.toTrackingProtectionException() })
}
}

private fun List<ContentPermission>?.filterTrackingProtectionExceptions() =
this.orEmpty().filter { it.isExcluded }

private val ContentPermission.isExcluded: Boolean
get() = this.permission == PERMISSION_TRACKING && value == VALUE_ALLOW

override fun add(session: EngineSession) {
val geckoEngineSession = (session as GeckoEngineSession)
runtime.contentBlockingController.addException(geckoEngineSession.geckoSession)
geckoEngineSession.notifyObservers {
onExcludedOnTrackingProtectionChange(true)
}
persist()
}

override fun remove(session: EngineSession) {
val geckoEngineSession = (session as GeckoEngineSession)
runtime.contentBlockingController.removeException(geckoEngineSession.geckoSession)
val url = geckoEngineSession.currentUrl ?: return
geckoEngineSession.notifyObservers {
onExcludedOnTrackingProtectionChange(false)
}
persist()
remove(url)
}

override fun remove(exception: TrackingProtectionException) {
val geckoException = (exception as GeckoTrackingProtectionException)
runtime.contentBlockingController.removeException(geckoException.toContentBlockingException())
persist()
if (exception is GeckoTrackingProtectionException) {
remove(exception.contentPermission)
} else {
remove(exception.url)
}
}

@VisibleForTesting
internal fun remove(url: String) {
val storage = runtime.storageController
storage.getPermissions(url).accept { permissions ->
permissions.filterTrackingProtectionExceptions().forEach { geckoPermissions ->
storage.setPermission(geckoPermissions, VALUE_DENY)
}
}
}

@VisibleForTesting
internal fun remove(contentPermission: ContentPermission) {
runtime.storageController.setPermission(contentPermission, VALUE_DENY)
}

override fun removeAll(activeSessions: List<EngineSession>?) {
runtime.contentBlockingController.clearExceptionList()
val storage = runtime.storageController
activeSessions?.forEach { engineSession ->
engineSession.notifyObservers {
onExcludedOnTrackingProtectionChange(false)
}
}
removeFileFromDisk(context)
storage.allPermissions.accept { permissions ->
val trackingExceptions = permissions.filterTrackingProtectionExceptions()
trackingExceptions.forEach {
storage.setPermission(it, VALUE_DENY)
}
}
}

@VisibleForTesting(otherwise = VisibleForTesting.PRIVATE)
internal fun getFile(context: Context): AtomicFile {
return AtomicFile(
File(
context.filesDir,
STORE_FILE_NAME_FORMAT
STORE_FILE_NAME
)
)
}

@VisibleForTesting(otherwise = VisibleForTesting.PRIVATE)
internal fun isMigrationOver(context: Context): Boolean {
/*
* We only keep around the exceptions file until
* the migration is over [STORE_FILE_NAME],
* after we migrate exceptions we delete the file.
* */
return File(context.filesDir, STORE_FILE_NAME).exists()
}

/**
* Take all the exception from the gecko [runtime] and saves them into the
* [STORE_FILE_NAME_FORMAT] file.
* As part of the migration process, we have to load all exceptions from our
* file [STORE_FILE_NAME] into geckoView once, after that we can remove our,
* file [STORE_FILE_NAME].
*/
private fun persist() {
runtime.contentBlockingController.saveExceptionList().accept { exceptionList ->
if (exceptionList != null) {
scope.launch {
synchronized(fileLock) {
getFile(context).writeString {
val jsonList = exceptionList.map { item ->
item.toJson()
}
JSONArray(jsonList).toString()
internal fun migrateExceptions() {
scope.launch {
synchronized(fileLock) {
getFile(context).readAndDeserialize { json ->
if (json.isNotEmpty()) {
val jsonArray = JSONArray(json)
val exceptionList = (0 until jsonArray.length()).map { index ->
val jsonObject = jsonArray.getJSONObject(index)
ContentBlockingException.fromJson(jsonObject)
}
runtime.contentBlockingController.restoreExceptionList(exceptionList)
}
}
} else {
removeFileFromDisk(context)
logger.debug("Finished tracking protection exceptions migration")
}
}
}

private fun removeFileFromDisk(context: Context) {
@VisibleForTesting
internal fun removeFileFromDisk(context: Context) {
scope.launch {
synchronized(fileLock) {
getFile(context)
Expand All @@ -156,16 +185,6 @@ internal class TrackingProtectionExceptionFileStorage(
}
}

private fun ContentBlockingException.toTrackingProtectionException(): GeckoTrackingProtectionException {
val json = toJson()
val principal = json.getString("principal")
val uri = json.getString("uri")
return GeckoTrackingProtectionException(uri, principal)
}

private fun GeckoTrackingProtectionException.toContentBlockingException(): ContentBlockingException {
val json = JSONObject()
json.put("principal", principal)
json.put("uri", url)
return ContentBlockingException.fromJson(json)
private fun ContentPermission.toTrackingProtectionException(): GeckoTrackingProtectionException {
return GeckoTrackingProtectionException(uri, privateMode, this)
}
Original file line number Diff line number Diff line change
Expand Up @@ -5,11 +5,16 @@
package mozilla.components.browser.engine.gecko.content.blocking

import mozilla.components.concept.engine.content.blocking.TrackingProtectionException
import org.mozilla.geckoview.GeckoSession.PermissionDelegate.ContentPermission

/**
* Represents a site that will be ignored by the tracking protection policies.
* @property url The url of the site to be ignored.
* @property principal Internal gecko identifier of an URI.
* @property privateMode Indicates if this exception should persisted in private mode.
* @property contentPermission The associated gecko content permission of this exception.
*/
data class GeckoTrackingProtectionException(override val url: String, val principal: String = "") :
TrackingProtectionException
data class GeckoTrackingProtectionException(
override val url: String,
val privateMode: Boolean = false,
val contentPermission: ContentPermission
) : TrackingProtectionException
Loading

0 comments on commit 7e25272

Please sign in to comment.