diff --git a/app/build.gradle.kts b/app/build.gradle.kts index 211d1ca5..354a2ccd 100644 --- a/app/build.gradle.kts +++ b/app/build.gradle.kts @@ -161,7 +161,7 @@ dependencies { implementation(libs.revanced.library) // Downloader plugins - implementation(project(":downloader-plugin")) + implementation(libs.plugin.api) // Native processes implementation(libs.kotlin.process) diff --git a/build.gradle.kts b/build.gradle.kts index 8ed32e02..c543b2fa 100644 --- a/build.gradle.kts +++ b/build.gradle.kts @@ -7,10 +7,4 @@ plugins { alias(libs.plugins.kotlin.parcelize) apply false alias(libs.plugins.about.libraries) apply false alias(libs.plugins.compose.compiler) apply false - alias(libs.plugins.binary.compatibility.validator) } - -apiValidation { - ignoredProjects.addAll(listOf("app", "example-downloader-plugin")) - nonPublicMarkers += "app.revanced.manager.plugin.downloader.PluginHostApi" -} \ No newline at end of file diff --git a/downloader-plugin/.gitignore b/downloader-plugin/.gitignore deleted file mode 100644 index 42afabfd..00000000 --- a/downloader-plugin/.gitignore +++ /dev/null @@ -1 +0,0 @@ -/build \ No newline at end of file diff --git a/downloader-plugin/api/downloader-plugin.api b/downloader-plugin/api/downloader-plugin.api deleted file mode 100644 index d3a22653..00000000 --- a/downloader-plugin/api/downloader-plugin.api +++ /dev/null @@ -1,171 +0,0 @@ -public abstract interface class app/revanced/manager/plugin/downloader/BaseDownloadScope : app/revanced/manager/plugin/downloader/Scope { -} - -public final class app/revanced/manager/plugin/downloader/ConstantsKt { - public static final field PLUGIN_HOST_PERMISSION Ljava/lang/String; -} - -public final class app/revanced/manager/plugin/downloader/DownloadUrl : android/os/Parcelable { - public static final field CREATOR Landroid/os/Parcelable$Creator; - public fun (Ljava/lang/String;Ljava/util/Map;)V - public synthetic fun (Ljava/lang/String;Ljava/util/Map;ILkotlin/jvm/internal/DefaultConstructorMarker;)V - public final fun component1 ()Ljava/lang/String; - public final fun component2 ()Ljava/util/Map; - public final fun copy (Ljava/lang/String;Ljava/util/Map;)Lapp/revanced/manager/plugin/downloader/DownloadUrl; - public static synthetic fun copy$default (Lapp/revanced/manager/plugin/downloader/DownloadUrl;Ljava/lang/String;Ljava/util/Map;ILjava/lang/Object;)Lapp/revanced/manager/plugin/downloader/DownloadUrl; - public final fun describeContents ()I - public fun equals (Ljava/lang/Object;)Z - public final fun getHeaders ()Ljava/util/Map; - public final fun getUrl ()Ljava/lang/String; - public fun hashCode ()I - public final fun toDownloadResult ()Lkotlin/Pair; - public fun toString ()Ljava/lang/String; - public final fun writeToParcel (Landroid/os/Parcel;I)V -} - -public final class app/revanced/manager/plugin/downloader/DownloadUrl$Creator : android/os/Parcelable$Creator { - public fun ()V - public final fun createFromParcel (Landroid/os/Parcel;)Lapp/revanced/manager/plugin/downloader/DownloadUrl; - public synthetic fun createFromParcel (Landroid/os/Parcel;)Ljava/lang/Object; - public final fun newArray (I)[Lapp/revanced/manager/plugin/downloader/DownloadUrl; - public synthetic fun newArray (I)[Ljava/lang/Object; -} - -public final class app/revanced/manager/plugin/downloader/Downloader { -} - -public final class app/revanced/manager/plugin/downloader/DownloaderBuilder { -} - -public final class app/revanced/manager/plugin/downloader/DownloaderKt { - public static final fun Downloader (Lkotlin/jvm/functions/Function1;)Lapp/revanced/manager/plugin/downloader/DownloaderBuilder; -} - -public final class app/revanced/manager/plugin/downloader/DownloaderScope : app/revanced/manager/plugin/downloader/Scope { - public final fun download (Lkotlin/jvm/functions/Function3;)V - public final fun get (Lkotlin/jvm/functions/Function4;)V - public fun getHostPackageName ()Ljava/lang/String; - public fun getPluginPackageName ()Ljava/lang/String; - public final fun useService (Landroid/content/Intent;Lkotlin/jvm/functions/Function2;Lkotlin/coroutines/Continuation;)Ljava/lang/Object; -} - -public final class app/revanced/manager/plugin/downloader/ExtensionsKt { - public static final fun download (Lapp/revanced/manager/plugin/downloader/DownloaderScope;Lkotlin/jvm/functions/Function4;)V -} - -public abstract interface class app/revanced/manager/plugin/downloader/GetScope : app/revanced/manager/plugin/downloader/Scope { - public abstract fun requestStartActivity (Landroid/content/Intent;Lkotlin/coroutines/Continuation;)Ljava/lang/Object; -} - -public abstract interface class app/revanced/manager/plugin/downloader/InputDownloadScope : app/revanced/manager/plugin/downloader/BaseDownloadScope { -} - -public abstract interface class app/revanced/manager/plugin/downloader/OutputDownloadScope : app/revanced/manager/plugin/downloader/BaseDownloadScope { - public abstract fun reportSize (JLkotlin/coroutines/Continuation;)Ljava/lang/Object; -} - -public final class app/revanced/manager/plugin/downloader/Package : android/os/Parcelable { - public static final field CREATOR Landroid/os/Parcelable$Creator; - public fun (Ljava/lang/String;Ljava/lang/String;)V - public final fun component1 ()Ljava/lang/String; - public final fun component2 ()Ljava/lang/String; - public final fun copy (Ljava/lang/String;Ljava/lang/String;)Lapp/revanced/manager/plugin/downloader/Package; - public static synthetic fun copy$default (Lapp/revanced/manager/plugin/downloader/Package;Ljava/lang/String;Ljava/lang/String;ILjava/lang/Object;)Lapp/revanced/manager/plugin/downloader/Package; - public final fun describeContents ()I - public fun equals (Ljava/lang/Object;)Z - public final fun getName ()Ljava/lang/String; - public final fun getVersion ()Ljava/lang/String; - public fun hashCode ()I - public fun toString ()Ljava/lang/String; - public final fun writeToParcel (Landroid/os/Parcel;I)V -} - -public final class app/revanced/manager/plugin/downloader/Package$Creator : android/os/Parcelable$Creator { - public fun ()V - public final fun createFromParcel (Landroid/os/Parcel;)Lapp/revanced/manager/plugin/downloader/Package; - public synthetic fun createFromParcel (Landroid/os/Parcel;)Ljava/lang/Object; - public final fun newArray (I)[Lapp/revanced/manager/plugin/downloader/Package; - public synthetic fun newArray (I)[Ljava/lang/Object; -} - -public abstract interface annotation class app/revanced/manager/plugin/downloader/PluginHostApi : java/lang/annotation/Annotation { -} - -public abstract interface class app/revanced/manager/plugin/downloader/Scope { - public abstract fun getHostPackageName ()Ljava/lang/String; - public abstract fun getPluginPackageName ()Ljava/lang/String; -} - -public abstract class app/revanced/manager/plugin/downloader/UserInteractionException : java/lang/Exception { - public synthetic fun (Ljava/lang/String;Lkotlin/jvm/internal/DefaultConstructorMarker;)V -} - -public abstract class app/revanced/manager/plugin/downloader/UserInteractionException$Activity : app/revanced/manager/plugin/downloader/UserInteractionException { - public synthetic fun (Ljava/lang/String;Lkotlin/jvm/internal/DefaultConstructorMarker;)V -} - -public final class app/revanced/manager/plugin/downloader/UserInteractionException$Activity$Cancelled : app/revanced/manager/plugin/downloader/UserInteractionException$Activity { -} - -public final class app/revanced/manager/plugin/downloader/UserInteractionException$Activity$NotCompleted : app/revanced/manager/plugin/downloader/UserInteractionException$Activity { - public final fun getIntent ()Landroid/content/Intent; - public final fun getResultCode ()I -} - -public final class app/revanced/manager/plugin/downloader/UserInteractionException$RequestDenied : app/revanced/manager/plugin/downloader/UserInteractionException { -} - -public final class app/revanced/manager/plugin/downloader/webview/APIKt { - public static final fun WebViewDownloader (Lkotlin/jvm/functions/Function4;)Lapp/revanced/manager/plugin/downloader/DownloaderBuilder; - public static final fun runWebView (Lapp/revanced/manager/plugin/downloader/GetScope;Ljava/lang/String;Lkotlin/jvm/functions/Function2;Lkotlin/coroutines/Continuation;)Ljava/lang/Object; -} - -public class app/revanced/manager/plugin/downloader/webview/IWebView$Default : app/revanced/manager/plugin/downloader/webview/IWebView { - public fun ()V - public fun asBinder ()Landroid/os/IBinder; - public fun finish ()V - public fun load (Ljava/lang/String;)V -} - -public abstract class app/revanced/manager/plugin/downloader/webview/IWebView$Stub : android/os/Binder, app/revanced/manager/plugin/downloader/webview/IWebView { - public fun ()V - public fun asBinder ()Landroid/os/IBinder; - public static fun asInterface (Landroid/os/IBinder;)Lapp/revanced/manager/plugin/downloader/webview/IWebView; - public fun onTransact (ILandroid/os/Parcel;Landroid/os/Parcel;I)Z -} - -public class app/revanced/manager/plugin/downloader/webview/IWebViewEvents$Default : app/revanced/manager/plugin/downloader/webview/IWebViewEvents { - public fun ()V - public fun asBinder ()Landroid/os/IBinder; - public fun download (Ljava/lang/String;Ljava/lang/String;Ljava/lang/String;)V - public fun pageLoad (Ljava/lang/String;)V - public fun ready (Lapp/revanced/manager/plugin/downloader/webview/IWebView;)V -} - -public abstract class app/revanced/manager/plugin/downloader/webview/IWebViewEvents$Stub : android/os/Binder, app/revanced/manager/plugin/downloader/webview/IWebViewEvents { - public fun ()V - public fun asBinder ()Landroid/os/IBinder; - public static fun asInterface (Landroid/os/IBinder;)Lapp/revanced/manager/plugin/downloader/webview/IWebViewEvents; - public fun onTransact (ILandroid/os/Parcel;Landroid/os/Parcel;I)Z -} - -public final class app/revanced/manager/plugin/downloader/webview/WebViewActivity$Parameters$Creator : android/os/Parcelable$Creator { - public fun ()V - public final fun createFromParcel (Landroid/os/Parcel;)Lapp/revanced/manager/plugin/downloader/webview/WebViewActivity$Parameters; - public synthetic fun createFromParcel (Landroid/os/Parcel;)Ljava/lang/Object; - public final fun newArray (I)[Lapp/revanced/manager/plugin/downloader/webview/WebViewActivity$Parameters; - public synthetic fun newArray (I)[Ljava/lang/Object; -} - -public abstract interface class app/revanced/manager/plugin/downloader/webview/WebViewCallbackScope : app/revanced/manager/plugin/downloader/Scope { - public abstract fun finish (Ljava/lang/Object;Lkotlin/coroutines/Continuation;)Ljava/lang/Object; - public abstract fun load (Ljava/lang/String;Lkotlin/coroutines/Continuation;)Ljava/lang/Object; -} - -public final class app/revanced/manager/plugin/downloader/webview/WebViewScope : app/revanced/manager/plugin/downloader/Scope { - public final fun download (Lkotlin/jvm/functions/Function5;)V - public fun getHostPackageName ()Ljava/lang/String; - public fun getPluginPackageName ()Ljava/lang/String; - public final fun pageLoad (Lkotlin/jvm/functions/Function3;)V -} - diff --git a/downloader-plugin/build.gradle.kts b/downloader-plugin/build.gradle.kts deleted file mode 100644 index 9d66a6e0..00000000 --- a/downloader-plugin/build.gradle.kts +++ /dev/null @@ -1,61 +0,0 @@ -plugins { - alias(libs.plugins.android.library) - alias(libs.plugins.kotlin.android) - alias(libs.plugins.kotlin.parcelize) - `maven-publish` -} - -android { - namespace = "app.revanced.manager.plugin.downloader" - compileSdk = 35 - - defaultConfig { - minSdk = 26 - - consumerProguardFiles("consumer-rules.pro") - } - - buildTypes { - release { - isMinifyEnabled = false - proguardFiles( - getDefaultProguardFile("proguard-android-optimize.txt"), - "proguard-rules.pro" - ) - } - } - compileOptions { - sourceCompatibility = JavaVersion.VERSION_17 - targetCompatibility = JavaVersion.VERSION_17 - } - kotlinOptions { - jvmTarget = "17" - } - buildFeatures { - aidl = true - } -} -dependencies { - implementation(libs.androidx.ktx) - implementation(libs.activity.ktx) - implementation(libs.runtime.ktx) - implementation(libs.appcompat) -} - -publishing { - repositories { - mavenLocal() - } - - publications { - create("release") { - groupId = "app.revanced" - artifactId = "manager-downloader-plugin" - version = "1.0" - - afterEvaluate { - from(components["release"]) - } - } - } -} \ No newline at end of file diff --git a/downloader-plugin/consumer-rules.pro b/downloader-plugin/consumer-rules.pro deleted file mode 100644 index e69de29b..00000000 diff --git a/downloader-plugin/proguard-rules.pro b/downloader-plugin/proguard-rules.pro deleted file mode 100644 index 481bb434..00000000 --- a/downloader-plugin/proguard-rules.pro +++ /dev/null @@ -1,21 +0,0 @@ -# Add project specific ProGuard rules here. -# You can control the set of applied configuration files using the -# proguardFiles setting in build.gradle. -# -# For more details, see -# http://developer.android.com/guide/developing/tools/proguard.html - -# If your project uses WebView with JS, uncomment the following -# and specify the fully qualified class name to the JavaScript interface -# class: -#-keepclassmembers class fqcn.of.javascript.interface.for.webview { -# public *; -#} - -# Uncomment this to preserve the line number information for -# debugging stack traces. -#-keepattributes SourceFile,LineNumberTable - -# If you keep the line number information, uncomment this to -# hide the original source file name. -#-renamesourcefileattribute SourceFile \ No newline at end of file diff --git a/downloader-plugin/src/main/AndroidManifest.xml b/downloader-plugin/src/main/AndroidManifest.xml deleted file mode 100644 index 74b7379f..00000000 --- a/downloader-plugin/src/main/AndroidManifest.xml +++ /dev/null @@ -1,3 +0,0 @@ - - - \ No newline at end of file diff --git a/downloader-plugin/src/main/aidl/app/revanced/manager/plugin/downloader/webview/IWebView.aidl b/downloader-plugin/src/main/aidl/app/revanced/manager/plugin/downloader/webview/IWebView.aidl deleted file mode 100644 index d657fcc3..00000000 --- a/downloader-plugin/src/main/aidl/app/revanced/manager/plugin/downloader/webview/IWebView.aidl +++ /dev/null @@ -1,8 +0,0 @@ -// IWebView.aidl -package app.revanced.manager.plugin.downloader.webview; - -@JavaPassthrough(annotation="@app.revanced.manager.plugin.downloader.PluginHostApi") -oneway interface IWebView { - void load(String url); - void finish(); -} \ No newline at end of file diff --git a/downloader-plugin/src/main/aidl/app/revanced/manager/plugin/downloader/webview/IWebViewEvents.aidl b/downloader-plugin/src/main/aidl/app/revanced/manager/plugin/downloader/webview/IWebViewEvents.aidl deleted file mode 100644 index b0237de2..00000000 --- a/downloader-plugin/src/main/aidl/app/revanced/manager/plugin/downloader/webview/IWebViewEvents.aidl +++ /dev/null @@ -1,11 +0,0 @@ -// IWebViewEvents.aidl -package app.revanced.manager.plugin.downloader.webview; - -import app.revanced.manager.plugin.downloader.webview.IWebView; - -@JavaPassthrough(annotation="@app.revanced.manager.plugin.downloader.PluginHostApi") -oneway interface IWebViewEvents { - void ready(IWebView iface); - void pageLoad(String url); - void download(String url, String mimetype, String userAgent); -} \ No newline at end of file diff --git a/downloader-plugin/src/main/java/app/revanced/manager/plugin/downloader/Constants.kt b/downloader-plugin/src/main/java/app/revanced/manager/plugin/downloader/Constants.kt deleted file mode 100644 index 469daaae..00000000 --- a/downloader-plugin/src/main/java/app/revanced/manager/plugin/downloader/Constants.kt +++ /dev/null @@ -1,7 +0,0 @@ -package app.revanced.manager.plugin.downloader - -/** - * The permission ID of the special plugin host permission. Only ReVanced Manager will have this permission. - * Plugin UI activities and internal services can be protected using this permission. - */ -const val PLUGIN_HOST_PERMISSION = "app.revanced.manager.permission.PLUGIN_HOST" \ No newline at end of file diff --git a/downloader-plugin/src/main/java/app/revanced/manager/plugin/downloader/Downloader.kt b/downloader-plugin/src/main/java/app/revanced/manager/plugin/downloader/Downloader.kt deleted file mode 100644 index bf0a219b..00000000 --- a/downloader-plugin/src/main/java/app/revanced/manager/plugin/downloader/Downloader.kt +++ /dev/null @@ -1,165 +0,0 @@ -package app.revanced.manager.plugin.downloader - -import android.content.ComponentName -import android.content.Context -import android.content.Intent -import android.content.ServiceConnection -import android.os.IBinder -import android.app.Activity -import android.os.Parcelable -import kotlinx.coroutines.withTimeout -import java.io.InputStream -import java.io.OutputStream -import kotlin.coroutines.resume -import kotlin.coroutines.suspendCoroutine - -@RequiresOptIn( - level = RequiresOptIn.Level.ERROR, - message = "This API is only intended for plugin hosts, don't use it in a plugin.", -) -@Retention(AnnotationRetention.BINARY) -annotation class PluginHostApi - -/** - * The base interface for all DSL scopes. - */ -interface Scope { - /** - * The package name of ReVanced Manager. - */ - val hostPackageName: String - - /** - * The package name of the plugin. - */ - val pluginPackageName: String -} - -/** - * The scope of [DownloaderScope.get]. - */ -interface GetScope : Scope { - /** - * Ask the user to perform some required interaction in the activity specified by the provided [Intent]. - * This function returns normally with the resulting [Intent] when the activity finishes with code [Activity.RESULT_OK]. - * - * @throws UserInteractionException.RequestDenied User decided to skip this plugin. - * @throws UserInteractionException.Activity.Cancelled The activity was cancelled. - * @throws UserInteractionException.Activity.NotCompleted The activity finished with an unknown result code. - */ - suspend fun requestStartActivity(intent: Intent): Intent? -} - -interface BaseDownloadScope : Scope - -/** - * The scope for [DownloaderScope.download]. - */ -interface InputDownloadScope : BaseDownloadScope - -typealias Size = Long -typealias DownloadResult = Pair - -typealias Version = String -typealias GetResult = Pair - -class DownloaderScope internal constructor( - private val scopeImpl: Scope, - internal val context: Context -) : Scope by scopeImpl { - // Returning an InputStream is the primary way for plugins to implement the download function, but we also want to offer an OutputStream API since using InputStream might not be convenient in all cases. - // It is much easier to implement the main InputStream API on top of OutputStreams compared to doing it the other way around, which is why we are using OutputStream here. This detail is not visible to plugins. - internal var download: (suspend OutputDownloadScope.(T, OutputStream) -> Unit)? = null - internal var get: (suspend GetScope.(String, String?) -> GetResult?)? = null - private val inputDownloadScopeImpl = object : InputDownloadScope, Scope by scopeImpl {} - - /** - * Define the download block of the plugin. - */ - fun download(block: suspend InputDownloadScope.(data: T) -> DownloadResult) { - download = { app, outputStream -> - val (inputStream, size) = inputDownloadScopeImpl.block(app) - - inputStream.use { - if (size != null) reportSize(size) - it.copyTo(outputStream) - } - } - } - - /** - * Define the get block of the plugin. - * The block should return null if the app cannot be found. The version in the result must match the version argument unless it is null. - */ - fun get(block: suspend GetScope.(packageName: String, version: String?) -> GetResult?) { - get = block - } - - /** - * Utilize the service specified by the provided [Intent]. The service will be unbound when the scope ends. - */ - suspend fun useService(intent: Intent, block: suspend (IBinder) -> R): R { - var onBind: ((IBinder) -> Unit)? = null - val serviceConn = object : ServiceConnection { - override fun onServiceConnected(name: ComponentName?, service: IBinder?) = - onBind!!(service!!) - - override fun onServiceDisconnected(name: ComponentName?) {} - } - - return try { - val binder = withTimeout(10000L) { - suspendCoroutine { continuation -> - onBind = continuation::resume - context.bindService(intent, serviceConn, Context.BIND_AUTO_CREATE) - } - } - block(binder) - } finally { - onBind = null - context.unbindService(serviceConn) - } - } -} - -class DownloaderBuilder internal constructor(private val block: DownloaderScope.() -> Unit) { - @PluginHostApi - fun build(scopeImpl: Scope, context: Context) = - with(DownloaderScope(scopeImpl, context)) { - block() - - Downloader( - download = download!!, - get = get!! - ) - } -} - -class Downloader internal constructor( - @property:PluginHostApi val get: suspend GetScope.(packageName: String, version: String?) -> GetResult?, - @property:PluginHostApi val download: suspend OutputDownloadScope.(data: T, outputStream: OutputStream) -> Unit -) - -/** - * Define a downloader plugin. - */ -fun Downloader(block: DownloaderScope.() -> Unit) = DownloaderBuilder(block) - -/** - * @see GetScope.requestStartActivity - */ -sealed class UserInteractionException(message: String) : Exception(message) { - class RequestDenied @PluginHostApi constructor() : - UserInteractionException("Request denied by user") - - sealed class Activity(message: String) : UserInteractionException(message) { - class Cancelled @PluginHostApi constructor() : Activity("Interaction cancelled") - - /** - * @param resultCode The result code of the activity. - * @param intent The [Intent] of the activity. - */ - class NotCompleted @PluginHostApi constructor(val resultCode: Int, val intent: Intent?) : - Activity("Unexpected activity result code: $resultCode") - } -} \ No newline at end of file diff --git a/downloader-plugin/src/main/java/app/revanced/manager/plugin/downloader/Extensions.kt b/downloader-plugin/src/main/java/app/revanced/manager/plugin/downloader/Extensions.kt deleted file mode 100644 index a1e6bf79..00000000 --- a/downloader-plugin/src/main/java/app/revanced/manager/plugin/downloader/Extensions.kt +++ /dev/null @@ -1,42 +0,0 @@ -package app.revanced.manager.plugin.downloader - -import android.app.Activity -import android.app.Service -import android.content.Intent -import android.os.IBinder -import android.os.Parcelable -import java.io.OutputStream - -/** - * The scope of the [OutputStream] version of [DownloaderScope.download]. - */ -interface OutputDownloadScope : BaseDownloadScope { - suspend fun reportSize(size: Long) -} - -/** - * A replacement for [DownloaderScope.download] that uses [OutputStream]. - * The provided [OutputStream] does not need to be closed manually. - */ -fun DownloaderScope.download(block: suspend OutputDownloadScope.(T, OutputStream) -> Unit) { - download = block -} - -/** - * Performs [GetScope.requestStartActivity] with an [Intent] created using the type information of [ACTIVITY]. - * @see [GetScope.requestStartActivity] - */ -suspend inline fun GetScope.requestStartActivity() = - requestStartActivity( - Intent().apply { setClassName(pluginPackageName, ACTIVITY::class.qualifiedName!!) } - ) - -/** - * Performs [DownloaderScope.useService] with an [Intent] created using the type information of [SERVICE]. - * @see [DownloaderScope.useService] - */ -suspend inline fun DownloaderScope<*>.useService( - noinline block: suspend (IBinder) -> R -) = useService( - Intent().apply { setClassName(pluginPackageName, SERVICE::class.qualifiedName!!) }, block -) \ No newline at end of file diff --git a/downloader-plugin/src/main/java/app/revanced/manager/plugin/downloader/Parcelables.kt b/downloader-plugin/src/main/java/app/revanced/manager/plugin/downloader/Parcelables.kt deleted file mode 100644 index 414ad889..00000000 --- a/downloader-plugin/src/main/java/app/revanced/manager/plugin/downloader/Parcelables.kt +++ /dev/null @@ -1,39 +0,0 @@ -package app.revanced.manager.plugin.downloader - -import android.os.Parcelable -import kotlinx.parcelize.Parcelize -import java.net.HttpURLConnection -import java.net.URI - -/** - * A simple parcelable data class for storing a package name and version. - * This can be used as the data type for plugins that only need a name and version to implement their [DownloaderScope.download] function. - * - * @param name The package name. - * @param version The version. - */ -@Parcelize -data class Package(val name: String, val version: String) : Parcelable - -/** - * A data class for storing a download URL. - * - * @param url The download URL. - * @param headers The headers to use for the request. - */ -@Parcelize -data class DownloadUrl(val url: String, val headers: Map = emptyMap()) : Parcelable { - /** - * Converts this into a [DownloadResult]. - */ - fun toDownloadResult(): DownloadResult = with(URI.create(url).toURL().openConnection() as HttpURLConnection) { - useCaches = false - allowUserInteraction = false - headers.forEach(::setRequestProperty) - - connectTimeout = 10_000 - connect() - - inputStream to getHeaderField("Content-Length").toLong() - } -} \ No newline at end of file diff --git a/downloader-plugin/src/main/java/app/revanced/manager/plugin/downloader/webview/API.kt b/downloader-plugin/src/main/java/app/revanced/manager/plugin/downloader/webview/API.kt deleted file mode 100644 index 2e5034e1..00000000 --- a/downloader-plugin/src/main/java/app/revanced/manager/plugin/downloader/webview/API.kt +++ /dev/null @@ -1,176 +0,0 @@ -package app.revanced.manager.plugin.downloader.webview - -import android.content.Intent -import app.revanced.manager.plugin.downloader.DownloadUrl -import app.revanced.manager.plugin.downloader.DownloaderScope -import app.revanced.manager.plugin.downloader.GetScope -import app.revanced.manager.plugin.downloader.Scope -import app.revanced.manager.plugin.downloader.Downloader -import app.revanced.manager.plugin.downloader.PluginHostApi -import kotlinx.coroutines.CoroutineScope -import kotlinx.coroutines.Dispatchers -import kotlinx.coroutines.ExperimentalCoroutinesApi -import kotlinx.coroutines.cancelChildren -import kotlinx.coroutines.launch -import kotlinx.coroutines.supervisorScope -import kotlinx.coroutines.withContext -import kotlin.properties.Delegates - -typealias InitialUrl = String -typealias PageLoadCallback = suspend WebViewCallbackScope.(url: String) -> Unit -typealias DownloadCallback = suspend WebViewCallbackScope.(url: String, mimeType: String, userAgent: String) -> Unit - -interface WebViewCallbackScope : Scope { - /** - * Finishes the activity and returns the [result]. - */ - suspend fun finish(result: T) - - /** - * Tells the WebView to load the specified [url]. - */ - suspend fun load(url: String) -} - -@OptIn(PluginHostApi::class) -class WebViewScope internal constructor( - coroutineScope: CoroutineScope, - private val scopeImpl: Scope, - setResult: (T) -> Unit -) : Scope by scopeImpl { - private var onPageLoadCallback: PageLoadCallback = {} - private var onDownloadCallback: DownloadCallback = { _, _, _ -> } - - @OptIn(ExperimentalCoroutinesApi::class) - private val dispatcher = Dispatchers.Default.limitedParallelism(1) - private lateinit var webView: IWebView - internal lateinit var initialUrl: String - - internal val binder = object : IWebViewEvents.Stub() { - override fun ready(iface: IWebView?) { - coroutineScope.launch(dispatcher) { - webView = iface!!.also { - it.load(initialUrl) - } - } - } - - override fun pageLoad(url: String?) { - coroutineScope.launch(dispatcher) { onPageLoadCallback(callbackScope, url!!) } - } - - override fun download(url: String?, mimetype: String?, userAgent: String?) { - coroutineScope.launch(dispatcher) { - onDownloadCallback( - callbackScope, - url!!, - mimetype!!, - userAgent!! - ) - } - } - } - - private val callbackScope = object : WebViewCallbackScope, Scope by scopeImpl { - override suspend fun finish(result: T) { - setResult(result) - // Tell the WebViewActivity to finish - webView.let { withContext(Dispatchers.IO) { it.finish() } } - } - - override suspend fun load(url: String) { - webView.let { withContext(Dispatchers.IO) { it.load(url) } } - } - - } - - /** - * Called when the WebView attempts to download a file to disk. - */ - fun download(block: DownloadCallback) { - onDownloadCallback = block - } - - /** - * Called when the WebView finishes loading a page. - */ - fun pageLoad(block: PageLoadCallback) { - onPageLoadCallback = block - } -} - -@JvmInline -private value class Container(val value: U) - -/** - * Run a [android.webkit.WebView] Activity controlled by the provided code block. - * The activity will keep running until it is cancelled or an event handler calls [WebViewCallbackScope.finish]. - * The [block] defines the event handlers and returns the initial URL. - * - * @param title The string displayed in the action bar. - * @param block The control block. - */ -@OptIn(PluginHostApi::class) -suspend fun GetScope.runWebView( - title: String, - block: suspend WebViewScope.() -> InitialUrl -) = supervisorScope { - var result by Delegates.notNull>() - - val scope = WebViewScope(this@supervisorScope, this@runWebView) { result = Container(it) } - scope.initialUrl = scope.block() - - // Start the webview activity and wait until it finishes. - requestStartActivity(Intent().apply { - putExtra( - WebViewActivity.KEY, - WebViewActivity.Parameters(title, scope.binder) - ) - setClassName( - hostPackageName, - WebViewActivity::class.qualifiedName!! - ) - }) - - // Return the result and cancel any leftover coroutines. - coroutineContext.cancelChildren() - result.value -} - -/** - * Implement a downloader using [runWebView] and [DownloadUrl]. This function will automatically define a handler for download events unlike [runWebView]. - * Returning null inside the [block] is equivalent to returning null inside [DownloaderScope.get]. - * - * @see runWebView - */ -fun WebViewDownloader(block: suspend WebViewScope.(packageName: String, version: String?) -> InitialUrl?) = - Downloader { - val label = context.applicationInfo.loadLabel( - context.packageManager - ).toString() - - get { packageName, version -> - class ReturnNull : Exception() - - try { - runWebView(label) { - download { url, _, userAgent -> - finish( - DownloadUrl( - url, - mapOf("User-Agent" to userAgent) - ) - ) - } - - block(this@runWebView, packageName, version) ?: throw ReturnNull() - } to version - } catch (_: ReturnNull) { - null - } - } - - download { - it.toDownloadResult() - } - } \ No newline at end of file diff --git a/downloader-plugin/src/main/java/app/revanced/manager/plugin/downloader/webview/WebViewActivity.kt b/downloader-plugin/src/main/java/app/revanced/manager/plugin/downloader/webview/WebViewActivity.kt deleted file mode 100644 index aff01337..00000000 --- a/downloader-plugin/src/main/java/app/revanced/manager/plugin/downloader/webview/WebViewActivity.kt +++ /dev/null @@ -1,161 +0,0 @@ -package app.revanced.manager.plugin.downloader.webview - -import android.annotation.SuppressLint -import android.os.Bundle -import android.os.IBinder -import android.os.Parcelable -import android.view.MenuItem -import android.webkit.CookieManager -import android.webkit.WebSettings -import android.webkit.WebView -import android.webkit.WebViewClient -import androidx.activity.ComponentActivity -import androidx.activity.addCallback -import androidx.activity.enableEdgeToEdge -import androidx.activity.viewModels -import androidx.core.view.ViewCompat -import androidx.core.view.WindowInsetsCompat -import androidx.lifecycle.Lifecycle -import androidx.lifecycle.ViewModel -import androidx.lifecycle.lifecycleScope -import androidx.lifecycle.repeatOnLifecycle -import androidx.lifecycle.viewModelScope -import app.revanced.manager.plugin.downloader.PluginHostApi -import app.revanced.manager.plugin.downloader.R -import kotlinx.coroutines.channels.Channel -import kotlinx.coroutines.flow.receiveAsFlow -import kotlinx.coroutines.launch -import kotlinx.parcelize.Parcelize - -@OptIn(PluginHostApi::class) -@PluginHostApi -class WebViewActivity : ComponentActivity() { - @SuppressLint("SetJavaScriptEnabled") - override fun onCreate(savedInstanceState: Bundle?) { - super.onCreate(savedInstanceState) - - val vm by viewModels() - enableEdgeToEdge() - setContentView(R.layout.activity_webview) - - ViewCompat.setOnApplyWindowInsetsListener(findViewById(R.id.main)) { v, insets -> - val systemBars = insets.getInsets(WindowInsetsCompat.Type.systemBars()) - v.setPadding(systemBars.left, systemBars.top, systemBars.right, systemBars.bottom) - insets - } - val webView = findViewById(R.id.webview) - onBackPressedDispatcher.addCallback { - if (webView.canGoBack()) webView.goBack() - else cancelActivity() - } - - val params = intent.getParcelableExtra(KEY)!! - actionBar?.apply { - title = params.title - setHomeAsUpIndicator(android.R.drawable.ic_menu_close_clear_cancel) - setDisplayHomeAsUpEnabled(true) - } - - val events = IWebViewEvents.Stub.asInterface(params.events)!! - vm.setup(events) - - webView.apply { - settings.apply { - cacheMode = WebSettings.LOAD_NO_CACHE - allowContentAccess = false - domStorageEnabled = true - javaScriptEnabled = true - } - - webViewClient = vm.webViewClient - setDownloadListener { url, userAgent, _, mimetype, _ -> - vm.onDownload(url, mimetype, userAgent) - } - } - - lifecycleScope.launch { - repeatOnLifecycle(Lifecycle.State.STARTED) { - vm.commands.collect { - when (it) { - is WebViewModel.Command.Finish -> { - setResult(RESULT_OK) - finish() - } - - is WebViewModel.Command.Load -> webView.loadUrl(it.url) - } - } - } - } - } - - private fun cancelActivity() { - setResult(RESULT_CANCELED) - finish() - } - - override fun onOptionsItemSelected(item: MenuItem) = if (item.itemId == android.R.id.home) { - cancelActivity() - - true - } else super.onOptionsItemSelected(item) - - @Parcelize - internal class Parameters( - val title: String, val events: IBinder - ) : Parcelable - - internal companion object { - const val KEY = "params" - } -} - -@OptIn(PluginHostApi::class) -internal class WebViewModel : ViewModel() { - init { - CookieManager.getInstance().apply { - removeAllCookies(null) - setAcceptCookie(true) - } - } - - private val commandChannel = Channel() - val commands = commandChannel.receiveAsFlow() - - private var eventBinder: IWebViewEvents? = null - private val ctrlBinder = object : IWebView.Stub() { - override fun load(url: String?) { - viewModelScope.launch { - commandChannel.send(Command.Load(url!!)) - } - } - - override fun finish() { - viewModelScope.launch { - commandChannel.send(Command.Finish) - } - } - } - - val webViewClient = object : WebViewClient() { - override fun onPageFinished(view: WebView?, url: String?) { - super.onPageFinished(view, url) - eventBinder!!.pageLoad(url) - } - } - - fun onDownload(url: String, mimeType: String, userAgent: String) { - eventBinder!!.download(url, mimeType, userAgent) - } - - fun setup(binder: IWebViewEvents) { - if (eventBinder != null) return - eventBinder = binder - binder.ready(ctrlBinder) - } - - sealed interface Command { - data class Load(val url: String) : Command - data object Finish : Command - } -} \ No newline at end of file diff --git a/downloader-plugin/src/main/res/layout/activity_webview.xml b/downloader-plugin/src/main/res/layout/activity_webview.xml deleted file mode 100644 index 51f761d9..00000000 --- a/downloader-plugin/src/main/res/layout/activity_webview.xml +++ /dev/null @@ -1,11 +0,0 @@ - - - - - \ No newline at end of file diff --git a/downloader-plugin/src/main/res/values/strings.xml b/downloader-plugin/src/main/res/values/strings.xml deleted file mode 100644 index 73862c41..00000000 --- a/downloader-plugin/src/main/res/values/strings.xml +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/downloader-plugin/src/main/res/values/themes.xml b/downloader-plugin/src/main/res/values/themes.xml deleted file mode 100644 index 495cde8e..00000000 --- a/downloader-plugin/src/main/res/values/themes.xml +++ /dev/null @@ -1,7 +0,0 @@ - - - - \ No newline at end of file diff --git a/example-downloader-plugin/.gitignore b/example-downloader-plugin/.gitignore deleted file mode 100644 index 42afabfd..00000000 --- a/example-downloader-plugin/.gitignore +++ /dev/null @@ -1 +0,0 @@ -/build \ No newline at end of file diff --git a/example-downloader-plugin/build.gradle.kts b/example-downloader-plugin/build.gradle.kts deleted file mode 100644 index b480add9..00000000 --- a/example-downloader-plugin/build.gradle.kts +++ /dev/null @@ -1,53 +0,0 @@ -plugins { - alias(libs.plugins.android.application) - alias(libs.plugins.kotlin.android) - alias(libs.plugins.kotlin.parcelize) - alias(libs.plugins.compose.compiler) -} - -android { - val packageName = "app.revanced.manager.plugin.downloader.example" - - namespace = packageName - compileSdk = 35 - - defaultConfig { - applicationId = packageName - minSdk = 26 - targetSdk = 35 - versionCode = 1 - versionName = "1.0" - } - - buildTypes { - release { - isMinifyEnabled = false - proguardFiles( - getDefaultProguardFile("proguard-android-optimize.txt"), - "proguard-rules.pro" - ) - - if (project.hasProperty("signAsDebug")) { - signingConfig = signingConfigs.getByName("debug") - } - } - } - compileOptions { - sourceCompatibility = JavaVersion.VERSION_17 - targetCompatibility = JavaVersion.VERSION_17 - } - kotlinOptions { - jvmTarget = "17" - } - buildFeatures.compose = true -} - -dependencies { - implementation(libs.activity.compose) - implementation(platform(libs.compose.bom)) - implementation(libs.compose.ui) - implementation(libs.compose.ui.tooling) - implementation(libs.compose.material3) - - compileOnly(project(":downloader-plugin")) -} \ No newline at end of file diff --git a/example-downloader-plugin/proguard-rules.pro b/example-downloader-plugin/proguard-rules.pro deleted file mode 100644 index 481bb434..00000000 --- a/example-downloader-plugin/proguard-rules.pro +++ /dev/null @@ -1,21 +0,0 @@ -# Add project specific ProGuard rules here. -# You can control the set of applied configuration files using the -# proguardFiles setting in build.gradle. -# -# For more details, see -# http://developer.android.com/guide/developing/tools/proguard.html - -# If your project uses WebView with JS, uncomment the following -# and specify the fully qualified class name to the JavaScript interface -# class: -#-keepclassmembers class fqcn.of.javascript.interface.for.webview { -# public *; -#} - -# Uncomment this to preserve the line number information for -# debugging stack traces. -#-keepattributes SourceFile,LineNumberTable - -# If you keep the line number information, uncomment this to -# hide the original source file name. -#-renamesourcefileattribute SourceFile \ No newline at end of file diff --git a/example-downloader-plugin/src/main/AndroidManifest.xml b/example-downloader-plugin/src/main/AndroidManifest.xml deleted file mode 100644 index e10b2d28..00000000 --- a/example-downloader-plugin/src/main/AndroidManifest.xml +++ /dev/null @@ -1,23 +0,0 @@ - - - - - - - - - - - - - \ No newline at end of file diff --git a/example-downloader-plugin/src/main/java/app/revanced/manager/plugin/downloader/example/ExamplePlugin.kt b/example-downloader-plugin/src/main/java/app/revanced/manager/plugin/downloader/example/ExamplePlugin.kt deleted file mode 100644 index dd2b26c5..00000000 --- a/example-downloader-plugin/src/main/java/app/revanced/manager/plugin/downloader/example/ExamplePlugin.kt +++ /dev/null @@ -1,69 +0,0 @@ -@file:Suppress("Unused") - -package app.revanced.manager.plugin.downloader.example - -import android.app.Application -import android.content.pm.PackageManager -import android.net.Uri -import android.os.Parcelable -import app.revanced.manager.plugin.downloader.Downloader -import app.revanced.manager.plugin.downloader.requestStartActivity -import app.revanced.manager.plugin.downloader.webview.WebViewDownloader -import kotlinx.parcelize.Parcelize -import kotlin.io.path.* - -val apkMirrorDownloader = WebViewDownloader { packageName, version -> - with(Uri.Builder()) { - scheme("https") - authority("www.apkmirror.com") - mapOf( - "post_type" to "app_release", - "searchtype" to "apk", - "s" to (version?.let { "$packageName $it" } ?: packageName), - "bundles%5B%5D" to "apk_files" // bundles[] - ).forEach { (key, value) -> - appendQueryParameter(key, value) - } - - build().toString() - } -} - -@Parcelize -class InstalledApp(val path: String) : Parcelable - -private val application by lazy { - // Don't do this in a real plugin. - val clazz = Class.forName("android.app.ActivityThread") - val activityThread = clazz.getMethod("currentActivityThread")(null) - clazz.getMethod("getApplication")(activityThread) as Application -} - -val installedAppDownloader = Downloader { - val pm = application.packageManager - - get { packageName, version -> - val packageInfo = try { - pm.getPackageInfo(packageName, 0) - } catch (_: PackageManager.NameNotFoundException) { - return@get null - } - if (version != null && packageInfo.versionName != version) return@get null - - requestStartActivity() - - InstalledApp(packageInfo.applicationInfo!!.sourceDir) to packageInfo.versionName - } - - - download { app -> - with(Path(app.path)) { inputStream() to fileSize() } - } - - /* - download { app, outputStream -> - val path = Path(app.path) - reportSize(path.fileSize()) - Files.copy(path, outputStream) - }*/ -} diff --git a/example-downloader-plugin/src/main/java/app/revanced/manager/plugin/downloader/example/InteractionActivity.kt b/example-downloader-plugin/src/main/java/app/revanced/manager/plugin/downloader/example/InteractionActivity.kt deleted file mode 100644 index 0390f3bd..00000000 --- a/example-downloader-plugin/src/main/java/app/revanced/manager/plugin/downloader/example/InteractionActivity.kt +++ /dev/null @@ -1,65 +0,0 @@ -package app.revanced.manager.plugin.downloader.example - -import android.os.Bundle -import androidx.activity.ComponentActivity -import androidx.activity.compose.setContent -import androidx.compose.foundation.isSystemInDarkTheme -import androidx.compose.foundation.layout.Column -import androidx.compose.foundation.layout.Row -import androidx.compose.foundation.layout.padding -import androidx.compose.material3.ExperimentalMaterial3Api -import androidx.compose.material3.MaterialTheme -import androidx.compose.material3.Scaffold -import androidx.compose.material3.Text -import androidx.compose.material3.TextButton -import androidx.compose.material3.TopAppBar -import androidx.compose.material3.darkColorScheme -import androidx.compose.material3.lightColorScheme -import androidx.compose.ui.Modifier - -class InteractionActivity : ComponentActivity() { - @OptIn(ExperimentalMaterial3Api::class) - override fun onCreate(savedInstanceState: Bundle?) { - super.onCreate(savedInstanceState) - - setContent { - val isDarkTheme = isSystemInDarkTheme() - - val colorScheme = if (isDarkTheme) darkColorScheme() else lightColorScheme() - - MaterialTheme(colorScheme) { - Scaffold( - topBar = { - TopAppBar( - title = { Text("User interaction example") } - ) - } - ) { paddingValues -> - Column(modifier = Modifier.padding(paddingValues)) { - Text("This is an example interaction.") - Row { - TextButton( - onClick = { - setResult(RESULT_CANCELED) - finish() - } - ) { - Text("Cancel") - } - - TextButton( - onClick = { - setResult(RESULT_OK) - finish() - } - ) { - Text("Continue") - } - } - } - } - } - - } - } -} \ No newline at end of file diff --git a/example-downloader-plugin/src/main/res/drawable/ic_launcher_background.xml b/example-downloader-plugin/src/main/res/drawable/ic_launcher_background.xml deleted file mode 100644 index 07d5da9c..00000000 --- a/example-downloader-plugin/src/main/res/drawable/ic_launcher_background.xml +++ /dev/null @@ -1,170 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/example-downloader-plugin/src/main/res/drawable/ic_launcher_foreground.xml b/example-downloader-plugin/src/main/res/drawable/ic_launcher_foreground.xml deleted file mode 100644 index 2b068d11..00000000 --- a/example-downloader-plugin/src/main/res/drawable/ic_launcher_foreground.xml +++ /dev/null @@ -1,30 +0,0 @@ - - - - - - - - - - - \ No newline at end of file diff --git a/example-downloader-plugin/src/main/res/mipmap-anydpi/ic_launcher.xml b/example-downloader-plugin/src/main/res/mipmap-anydpi/ic_launcher.xml deleted file mode 100644 index 6f3b755b..00000000 --- a/example-downloader-plugin/src/main/res/mipmap-anydpi/ic_launcher.xml +++ /dev/null @@ -1,6 +0,0 @@ - - - - - - \ No newline at end of file diff --git a/example-downloader-plugin/src/main/res/mipmap-anydpi/ic_launcher_round.xml b/example-downloader-plugin/src/main/res/mipmap-anydpi/ic_launcher_round.xml deleted file mode 100644 index 6f3b755b..00000000 --- a/example-downloader-plugin/src/main/res/mipmap-anydpi/ic_launcher_round.xml +++ /dev/null @@ -1,6 +0,0 @@ - - - - - - \ No newline at end of file diff --git a/example-downloader-plugin/src/main/res/values/strings.xml b/example-downloader-plugin/src/main/res/values/strings.xml deleted file mode 100644 index 4006549c..00000000 --- a/example-downloader-plugin/src/main/res/values/strings.xml +++ /dev/null @@ -1,3 +0,0 @@ - - Example Downloader Plugin - \ No newline at end of file diff --git a/gradle/libs.versions.toml b/gradle/libs.versions.toml index 70749da4..ffc6f295 100644 --- a/gradle/libs.versions.toml +++ b/gradle/libs.versions.toml @@ -19,6 +19,7 @@ datetime = "0.6.1" room-version = "2.6.1" revanced-patcher = "21.0.0" revanced-library = "3.0.2" +plugin-api = "1.0.0" koin = "3.5.3" ktor = "2.3.9" markdown-renderer = "0.30.0" @@ -27,7 +28,6 @@ kotlin = "2.1.10" android-gradle-plugin = "8.8.0" dev-tools-gradle-plugin = "2.1.10-1.0.29" about-libraries-gradle-plugin = "11.5.0" -binary-compatibility-validator = "0.17.0" coil = "2.7.0" app-icon-loader-coil = "1.5.0" skrapeit = "1.2.2" @@ -45,7 +45,6 @@ runtime-ktx = { group = "androidx.lifecycle", name = "lifecycle-runtime-ktx", ve runtime-compose = { group = "androidx.lifecycle", name = "lifecycle-runtime-compose", version.ref = "viewmodel-lifecycle" } splash-screen = { group = "androidx.core", name = "core-splashscreen", version.ref = "splash-screen" } activity-compose = { group = "androidx.activity", name = "activity-compose", version.ref = "activity" } -activity-ktx = { group = "androidx.activity", name = "activity-ktx", version.ref = "activity" } work-runtime-ktx = { group = "androidx.work", name = "work-runtime-ktx", version.ref = "work-runtime" } preferences-datastore = { group = "androidx.datastore", name = "datastore-preferences", version.ref = "preferences-datastore" } appcompat = { group = "androidx.appcompat", name = "appcompat", version.ref = "appcompat" } @@ -84,6 +83,9 @@ room-compiler = { group = "androidx.room", name = "room-compiler", version.ref = revanced-patcher = { group = "app.revanced", name = "revanced-patcher", version.ref = "revanced-patcher" } revanced-library = { group = "app.revanced", name = "revanced-library", version.ref = "revanced-library" } +# Plugin API +plugin-api = { group = "app.revanced", name = "revanced-manager-downloader-api", version.ref = "plugin-api" } + # Koin koin-android = { group = "io.insert-koin", name = "koin-android", version.ref = "koin" } koin-compose = { group = "io.insert-koin", name = "koin-androidx-compose", version.ref = "koin" } @@ -143,5 +145,4 @@ kotlin-serialization = { id = "org.jetbrains.kotlin.plugin.serialization", versi kotlin-parcelize = { id = "org.jetbrains.kotlin.plugin.parcelize", version.ref = "kotlin" } compose-compiler = { id = "org.jetbrains.kotlin.plugin.compose", version.ref = "kotlin" } devtools = { id = "com.google.devtools.ksp", version.ref = "dev-tools-gradle-plugin" } -about-libraries = { id = "com.mikepenz.aboutlibraries.plugin", version.ref = "about-libraries-gradle-plugin" } -binary-compatibility-validator = { id = "org.jetbrains.kotlinx.binary-compatibility-validator", version.ref = "binary-compatibility-validator" } \ No newline at end of file +about-libraries = { id = "com.mikepenz.aboutlibraries.plugin", version.ref = "about-libraries-gradle-plugin" } \ No newline at end of file diff --git a/settings.gradle.kts b/settings.gradle.kts index 18b2f456..f66506b8 100644 --- a/settings.gradle.kts +++ b/settings.gradle.kts @@ -26,5 +26,3 @@ dependencyResolutionManagement { } rootProject.name = "ReVanced Manager" include(":app") -include(":downloader-plugin") -include(":example-downloader-plugin")