From 687b884dc40874011986d280934511befe912bd4 Mon Sep 17 00:00:00 2001 From: oSumAtrIX Date: Wed, 14 Feb 2024 01:59:17 +0100 Subject: [PATCH 1/9] ci: Import GPG key as late as possible It is not necessary to import it earlier. --- .github/workflows/release.yml | 14 +++++++------- 1 file changed, 7 insertions(+), 7 deletions(-) diff --git a/.github/workflows/release.yml b/.github/workflows/release.yml index b6d81cdb..a1253603 100644 --- a/.github/workflows/release.yml +++ b/.github/workflows/release.yml @@ -24,13 +24,6 @@ jobs: persist-credentials: false fetch-depth: 0 - - name: Import GPG key - uses: crazy-max/ghaction-import-gpg@v6 - with: - gpg_private_key: ${{ secrets.GPG_PRIVATE_KEY }} - passphrase: ${{ secrets.PASSPHRASE }} - fingerprint: ${{ env.GPG_FINGERPRINT }} - - name: Cache Gradle uses: burrunan/gradle-cache-action@v1 @@ -48,6 +41,13 @@ jobs: - name: Install dependencies run: npm install + - name: Import GPG key + uses: crazy-max/ghaction-import-gpg@v6 + with: + gpg_private_key: ${{ secrets.GPG_PRIVATE_KEY }} + passphrase: ${{ secrets.PASSPHRASE }} + fingerprint: ${{ env.GPG_FINGERPRINT }} + - name: Release env: GITHUB_TOKEN: ${{ secrets.REPOSITORY_PUSH_ACCESS }} From f1c60093cfd2f2eccb112d2766e261b59839a9e5 Mon Sep 17 00:00:00 2001 From: oSumAtrIX Date: Wed, 14 Feb 2024 02:49:52 +0100 Subject: [PATCH 2/9] ci: Update GPG passphrase secret name --- .github/workflows/release.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/release.yml b/.github/workflows/release.yml index a1253603..9c384f38 100644 --- a/.github/workflows/release.yml +++ b/.github/workflows/release.yml @@ -45,7 +45,7 @@ jobs: uses: crazy-max/ghaction-import-gpg@v6 with: gpg_private_key: ${{ secrets.GPG_PRIVATE_KEY }} - passphrase: ${{ secrets.PASSPHRASE }} + passphrase: ${{ secrets.GPG_PASSPHRASE }} fingerprint: ${{ env.GPG_FINGERPRINT }} - name: Release From fe8ea9130d8a801e106438ade08264e4c79ac59f Mon Sep 17 00:00:00 2001 From: oSumAtrIX Date: Wed, 14 Feb 2024 04:32:18 +0100 Subject: [PATCH 3/9] build: Remove unnecessary `java` plugin --- build.gradle.kts | 6 ------ 1 file changed, 6 deletions(-) diff --git a/build.gradle.kts b/build.gradle.kts index 7c89ba39..4b41cf4f 100644 --- a/build.gradle.kts +++ b/build.gradle.kts @@ -3,7 +3,6 @@ plugins { alias(libs.plugins.binary.compatibility.validator) `maven-publish` signing - java } group = "app.revanced" @@ -52,11 +51,6 @@ dependencies { testImplementation(libs.kotlin.test) } -java { - withJavadocJar() - withSourcesJar() -} - kotlin { jvmToolchain(11) } From cc183062ab27fa30f7233882fb01fdd6707f566d Mon Sep 17 00:00:00 2001 From: oSumAtrIX Date: Thu, 15 Feb 2024 02:46:14 +0100 Subject: [PATCH 4/9] chore: Fix inline comment --- .../kotlin/app/revanced/patcher/patch/options/PatchOption.kt | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/src/main/kotlin/app/revanced/patcher/patch/options/PatchOption.kt b/src/main/kotlin/app/revanced/patcher/patch/options/PatchOption.kt index cc0d1303..d9409505 100644 --- a/src/main/kotlin/app/revanced/patcher/patch/options/PatchOption.kt +++ b/src/main/kotlin/app/revanced/patcher/patch/options/PatchOption.kt @@ -80,7 +80,6 @@ open class PatchOption( if (!validator(value)) throw PatchOptionException.ValueValidationException(value, this) } - override fun toString() = value.toString() operator fun getValue( @@ -439,7 +438,7 @@ open class PatchOption( ) /** - * Create a new [PatchOption] with a string set value and add it to the current [Patch]. + * Create a new [PatchOption] and add it to the current [Patch]. * * @param key The identifier. * @param default The default value. From b0b2c10665de2fb240614d2e1b0b401c165df718 Mon Sep 17 00:00:00 2001 From: oSumAtrIX Date: Thu, 22 Feb 2024 04:23:53 +0100 Subject: [PATCH 5/9] ci: Split release into a separate PR build workflow Because the release workflow already runs on dev and main, it is not necessary to also trigger it for PRs. --- .github/workflows/build_pull_request.yml | 25 +++++++++++++++++++ ...pull_request.yml => open_pull_request.yml} | 0 .github/workflows/release.yml | 4 --- 3 files changed, 25 insertions(+), 4 deletions(-) create mode 100644 .github/workflows/build_pull_request.yml rename .github/workflows/{pull_request.yml => open_pull_request.yml} (100%) diff --git a/.github/workflows/build_pull_request.yml b/.github/workflows/build_pull_request.yml new file mode 100644 index 00000000..28995022 --- /dev/null +++ b/.github/workflows/build_pull_request.yml @@ -0,0 +1,25 @@ +name: Build pull request + +on: + workflow_dispatch: + pull_request: + branches: + - dev + +jobs: + release: + name: Build + runs-on: ubuntu-latest + steps: + - name: Checkout + uses: actions/checkout@v4 + with: + fetch-depth: 0 + + - name: Cache Gradle + uses: burrunan/gradle-cache-action@v1 + + - name: Build + env: + GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} + run: ./gradlew build --no-daemon diff --git a/.github/workflows/pull_request.yml b/.github/workflows/open_pull_request.yml similarity index 100% rename from .github/workflows/pull_request.yml rename to .github/workflows/open_pull_request.yml diff --git a/.github/workflows/release.yml b/.github/workflows/release.yml index 9c384f38..2cb2b203 100644 --- a/.github/workflows/release.yml +++ b/.github/workflows/release.yml @@ -6,10 +6,6 @@ on: branches: - main - dev - pull_request: - branches: - - main - - dev jobs: release: From d0a57ac00da5faa163ac621262c92f6156886a50 Mon Sep 17 00:00:00 2001 From: oSumAtrIX Date: Fri, 23 Feb 2024 03:56:19 +0100 Subject: [PATCH 6/9] docs: Add readme and contribution guidelines --- CONTRIBUTING.md | 99 +++++++++++++++++++++++++++++++++++++++ README.md | 122 ++++++++++++++++++++++++++++++++++++++++++++++++ 2 files changed, 221 insertions(+) create mode 100644 CONTRIBUTING.md diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md new file mode 100644 index 00000000..e44be5a6 --- /dev/null +++ b/CONTRIBUTING.md @@ -0,0 +1,99 @@ +

+ + + + +
+ + + + + +     + + + + + +     + + + + + +     + + + + + +     + + + + + +     + + + + + +     + + + + + + +
+
+ Continuing the legacy of Vanced +

+ +# πŸ‘‹ Contribution guidelines + +This document describes how to contribute to ReVanced Patcher. + +## πŸ“– Resources to help you get started + +- The [documentation](https://github.com/ReVanced/revanced-patcher/tree/docs) contains the fundamentals + of ReVanced Patcher and how to use ReVanced Patcher to create patches +- [Our backlog](https://github.com/orgs/ReVanced/projects/12) is where we keep track of what we're working on +- [Issues](https://github.com/ReVanced/revanced-patcher/issues) are where we keep track of bugs and feature requests + +## πŸ™ Submitting a feature request + +Features can be requested by opening an issue using the +[Feature request issue template](https://github.com/ReVanced/revanced-patcher/issues/new?assignees=&labels=Feature+request&projects=&template=feature-request.yml&title=feat%3A+). + +> **Note** +> Requests can be accepted or rejected at the discretion of maintainers of ReVanced Patcher. +> Good motivation has to be provided for a request to be accepted. + +## 🐞 Submitting a bug report + +If you encounter a bug while using ReVanced Patcher, open an issue using the +[Bug report issue template](https://github.com/ReVanced/revanced-patcher/issues/new?assignees=&labels=Bug+report&projects=&template=bug-report.yml&title=bug%3A+). + +## πŸ“ How to contribute + +1. Before contributing, it is recommended to open an issue to discuss your change + with the maintainers of ReVanced Patcher. This will help you determine whether your change is acceptable + and whether it is worth your time to implement it +2. Development happens on the `dev` branch. Fork the repository and create your branch from `dev` +3. Commit your changes +4. Submit a pull request to the `dev` branch of the repository and reference issues + that your pull request closes in the description of your pull request +5. Our team will review your pull request and provide feedback. Once your pull request is approved, + it will be merged into the `dev` branch and will be included in the next release of ReVanced Patcher + +❀️ Thank you for considering contributing to ReVanced Patcher, +ReVanced diff --git a/README.md b/README.md index ed9ccf28..353926d5 100644 --- a/README.md +++ b/README.md @@ -1,3 +1,125 @@ +

+ + + + +
+ + + + + +     + + + + + +     + + + + + +     + + + + + +     + + + + + +     + + + + + +     + + + + + + +
+
+ Continuing the legacy of Vanced +

+ # πŸ’‰ ReVanced Patcher +![GitHub Workflow Status (with event)](https://img.shields.io/github/actions/workflow/status/ReVanced/revanced-patcher/release.yml) +![GPLv3 License](https://img.shields.io/badge/License-GPL%20v3-yellow.svg) + ReVanced Patcher used to patch Android applications. + +## ❓ About + +ReVanced Patcher is a library that is used to patch Android applications. +It powers [ReVanced Manager](https://github.com/ReVanced/revanced-manager), +[ReVanced CLI](https://github.com/ReVanced/revanced-cli) +and [ReVanced Library](https://github.com/ReVanced/revanced-library) and a rich set of patches have been developed +using ReVanced Patcher in the [ReVanced Patches](https://github.com/ReVanced/revanced-patches) repository. + +## πŸ’ͺ Features + +Some of the features the ReVanced Patcher provides are: + +- πŸ”§ **Patch Dalvik VM bytecode**: Disassemble and assemble Dalvik bytecode +- πŸ“¦ **Patch APK resources**: Decode and build Android APK resources +- πŸ“‚ **Patch arbitrary APK files**: Read and write arbitrary files directly from and to APK files +- 🧩 **Write modular patches**: Extensive API to write modular patches that can patch Dalvik VM bytecode, +APK resources and arbitrary APK files + +## πŸš€ How to get started + +To use ReVanced Patcher in your project, follow these steps: + +1. [Add the repository](https://docs.github.com/en/packages/working-with-a-github-packages-registry/working-with-the-gradle-registry#using-a-published-package) +to your project +2. Add the dependency to your project: + + ```kotlin + dependencies { + implementation("app.revanced:revanced-patcher:{$version}") + } + ``` + +For a minimal project configuration, +see [ReVanced Patches template](https://github.com/ReVanced/revanced-patches-template). + +## πŸ“š Everything else + +### πŸ“™ Contributing + +Thank you for considering contributing to ReVanced Patcher. +You can find the contribution guidelines [here](CONTRIBUTING.md). + +### πŸ› οΈ Building + +To build ReVanced Patcher, +you can follow the [ReVanced documentation](https://github.com/ReVanced/revanced-documentation). + +### πŸ“ƒ Documentation + +The documentation contains the fundamentals of ReVanced Patcher and how to use ReVanced Patcher to create patches. +You can find it [here](https://github.com/ReVanced/revanced-patcher/tree/docs). + +## πŸ“œ Licence + +ReVanced Patcher is licensed under the GPLv3 license. Please see the [licence file](LICENSE) for more information. +[tl;dr](https://www.tldrlegal.com/license/gnu-general-public-license-v3-gpl-3) you may copy, distribute and modify ReVanced Patcher as long as you track changes/dates in source files. +Any modifications to ReVanced Patcher must also be made available under the GPL, +along with build & install instructions. From 33ed5f0aa330940e3158be667875cd4e22bf36fb Mon Sep 17 00:00:00 2001 From: oSumAtrIX Date: Sat, 24 Feb 2024 01:14:43 +0100 Subject: [PATCH 7/9] ci: Fix indentation in workflow --- .github/workflows/build_pull_request.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/build_pull_request.yml b/.github/workflows/build_pull_request.yml index 28995022..250871bc 100644 --- a/.github/workflows/build_pull_request.yml +++ b/.github/workflows/build_pull_request.yml @@ -22,4 +22,4 @@ jobs: - name: Build env: GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} - run: ./gradlew build --no-daemon + run: ./gradlew build --no-daemon From 41257ee87ef9d3b520fc83bde7fada284c079f24 Mon Sep 17 00:00:00 2001 From: oSumAtrIX Date: Sun, 25 Feb 2024 03:26:15 +0100 Subject: [PATCH 8/9] chore: Rename issue templates and use relative image paths --- .../{bug-report.yml => bug_report.yml} | 14 +++++++------- .../{feature-request.yml => feature_request.yml} | 14 +++++++------- 2 files changed, 14 insertions(+), 14 deletions(-) rename .github/ISSUE_TEMPLATE/{bug-report.yml => bug_report.yml} (88%) rename .github/ISSUE_TEMPLATE/{feature-request.yml => feature_request.yml} (88%) diff --git a/.github/ISSUE_TEMPLATE/bug-report.yml b/.github/ISSUE_TEMPLATE/bug_report.yml similarity index 88% rename from .github/ISSUE_TEMPLATE/bug-report.yml rename to .github/ISSUE_TEMPLATE/bug_report.yml index 1dbbb1bf..8803c2d5 100644 --- a/.github/ISSUE_TEMPLATE/bug-report.yml +++ b/.github/ISSUE_TEMPLATE/bug_report.yml @@ -1,7 +1,7 @@ name: 🐞 Bug report description: Report a bug or an issue. -title: 'bug: ' -labels: ['Bug report'] +title: "bug: " +labels: ["Bug report"] body: - type: markdown attributes: @@ -11,18 +11,18 @@ body:
- - + +     @@ -69,7 +69,7 @@ body: # ReVanced Patcher bug report Before creating a new bug report, please keep the following in mind: - + - **Do not submit a duplicate bug report**: You can review existing bug reports [here](https://github.com/ReVanced/revanced-patcher/labels/Bug%20report). - **Do not use the issue page for support**: If you need help or have questions, check out other platforms on [revanced.app](https://revanced.app). - type: textarea diff --git a/.github/ISSUE_TEMPLATE/feature-request.yml b/.github/ISSUE_TEMPLATE/feature_request.yml similarity index 88% rename from .github/ISSUE_TEMPLATE/feature-request.yml rename to .github/ISSUE_TEMPLATE/feature_request.yml index 335e8dbb..e35a9e5a 100644 --- a/.github/ISSUE_TEMPLATE/feature-request.yml +++ b/.github/ISSUE_TEMPLATE/feature_request.yml @@ -1,7 +1,7 @@ name: ⭐ Feature request description: Create a detailed request for a new feature. -title: 'feat: ' -labels: ['Feature request'] +title: "feat: " +labels: ["Feature request"] body: - type: markdown attributes: @@ -11,18 +11,18 @@ body:
- - + +     @@ -85,7 +85,7 @@ body: label: Motivation description: | A strong motivation is necessary for a feature request to be considered. - + - Why should this feature be implemented? - What is the explicit use case? - What are the benefits? From fe616beb22b4e36b49f2b41b990155d37603bcc8 Mon Sep 17 00:00:00 2001 From: oSumAtrIX Date: Sun, 25 Feb 2024 03:30:08 +0100 Subject: [PATCH 9/9] docs: Add documentation (#278) --- .github/workflows/update_documentation.yml | 19 ++ CONTRIBUTING.md | 2 +- README.md | 4 +- docs/1_patcher_intro.md | 108 ++++++++ docs/2_1_setup.md | 107 ++++++++ docs/2_2_1_fingerprinting.md | 275 +++++++++++++++++++++ docs/2_2_patch_anatomy.md | 211 ++++++++++++++++ docs/2_patches_intro.md | 125 ++++++++++ docs/3_structure_and_conventions.md | 97 ++++++++ docs/4_apis.md | 23 ++ docs/README.md | 73 ++++++ 11 files changed, 1041 insertions(+), 3 deletions(-) create mode 100644 .github/workflows/update_documentation.yml create mode 100644 docs/1_patcher_intro.md create mode 100644 docs/2_1_setup.md create mode 100644 docs/2_2_1_fingerprinting.md create mode 100644 docs/2_2_patch_anatomy.md create mode 100644 docs/2_patches_intro.md create mode 100644 docs/3_structure_and_conventions.md create mode 100644 docs/4_apis.md create mode 100644 docs/README.md diff --git a/.github/workflows/update_documentation.yml b/.github/workflows/update_documentation.yml new file mode 100644 index 00000000..77097e2f --- /dev/null +++ b/.github/workflows/update_documentation.yml @@ -0,0 +1,19 @@ +name: Update documentation + +on: + push: + paths: + - docs/** + +jobs: + trigger: + runs-on: ubuntu-latest + name: Dispatch event to documentation repository + if: github.ref == 'refs/heads/main' + steps: + - uses: peter-evans/repository-dispatch@v2 + with: + token: ${{ secrets.DOCUMENTATION_REPO_ACCESS_TOKEN }} + repository: revanced/revanced-documentation + event-type: update-documentation + client-payload: '{"repo": "${{ github.event.repository.name }}", "ref": "${{ github.ref }}"}' diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md index e44be5a6..a33b3159 100644 --- a/CONTRIBUTING.md +++ b/CONTRIBUTING.md @@ -64,7 +64,7 @@ This document describes how to contribute to ReVanced Patcher. ## πŸ“– Resources to help you get started -- The [documentation](https://github.com/ReVanced/revanced-patcher/tree/docs) contains the fundamentals +- The [documentation](https://github.com/ReVanced/revanced-patcher/tree/docs/docs) contains the fundamentals of ReVanced Patcher and how to use ReVanced Patcher to create patches - [Our backlog](https://github.com/orgs/ReVanced/projects/12) is where we keep track of what we're working on - [Issues](https://github.com/ReVanced/revanced-patcher/issues) are where we keep track of bugs and feature requests diff --git a/README.md b/README.md index 353926d5..d30613e3 100644 --- a/README.md +++ b/README.md @@ -91,7 +91,7 @@ To use ReVanced Patcher in your project, follow these steps: to your project 2. Add the dependency to your project: - ```kotlin + ```kt dependencies { implementation("app.revanced:revanced-patcher:{$version}") } @@ -115,7 +115,7 @@ you can follow the [ReVanced documentation](https://github.com/ReVanced/revanced ### πŸ“ƒ Documentation The documentation contains the fundamentals of ReVanced Patcher and how to use ReVanced Patcher to create patches. -You can find it [here](https://github.com/ReVanced/revanced-patcher/tree/docs). +You can find it [here](https://github.com/ReVanced/revanced-patcher/tree/docs/docs). ## πŸ“œ Licence diff --git a/docs/1_patcher_intro.md b/docs/1_patcher_intro.md new file mode 100644 index 00000000..983d7582 --- /dev/null +++ b/docs/1_patcher_intro.md @@ -0,0 +1,108 @@ +

+ + + + +
+
+ + + + +     + + + + + +     + + + + + +     + + + + + +     + + + + + +     + + + + + +     + + + + + + +
+
+ Continuing the legacy of Vanced +

+ +# πŸ’‰ Introduction to ReVanced Patcher + +In order to create patches for Android applications, you first need to understand the fundamentals of ReVanced Patcher. + +## πŸ“™ How it works + +ReVanced Patcher is a library that allows you to modify Android applications by applying patches to their APKs. It is built on top of [Smali](https://github.com/google/smali) for bytecode manipulation and [Androlib (Apktool)](https://github.com/iBotPeaches/Apktool) for resource decoding and encoding. +ReVanced Patcher accepts a list of patches and integrations, and applies them to a given APK file. It then returns the modified components of the APK file, such as modified dex files and resources, that can be repackaged into a new APK file. + +ReVanced Patcher has a simple API that allows you to load patches and integrations from JAR files and apply them to an APK file. +Later on, you will learn how to create patches. + +```kt + // Executed patches do not necessarily reset their state. + // For that reason it is important to create a new instance of the PatchBundleLoader + // once the patches are executed instead of reusing the same instance of patches loaded by PatchBundleLoader. +val patches: PatchSet /* = Set> */ = PatchBundleLoader.Jar(File("revanced-patches.jar")) +val integrations = setOf(File("integrations.apk")) + +// Instantiating the patcher will decode the manifest of the APK file to read the package and version name. +val patcherConfig = PatcherConfig(apkFile = File("some.apk")) +val patcherResult = Patcher(patcherConfig).use { patcher -> + patcher.apply { + acceptIntegrations(integrations) + acceptPatches(patches) + + // Execute patches. + runBlocking { + patcher.apply(returnOnError = false).collect { patchResult -> + if (patchResult.exception != null) + println("${patchResult.patchName} failed:\n${patchResult.exception}") + else + println("${patchResult.patchName} succeeded") + } + } + }.get() +} + +// The result of the patcher contains the modified components of the APK file that can be repackaged into a new APK file. +val dexFiles = patcherResult.dexFiles +val resources = patcherResult.resources +``` + +## ⏭️ What's next + +The next page teaches the fundamentals of ReVanced Patches. + +Continue: [🧩 Introduction to ReVanced Patches](2_patches_intro.md) diff --git a/docs/2_1_setup.md b/docs/2_1_setup.md new file mode 100644 index 00000000..ae962aba --- /dev/null +++ b/docs/2_1_setup.md @@ -0,0 +1,107 @@ +

+ + + + +
+ + + + + +     + + + + + +     + + + + + +     + + + + + +     + + + + + +     + + + + + +     + + + + + + +
+
+ Continuing the legacy of Vanced +

+ +# πŸ‘Ά Setting up a development environment + +To get started developing patches with ReVanced Patcher, you need to prepare a development environment. + +## πŸ“ Prerequisites + +- A Java IDE with Kotlin support, such as [IntelliJ IDEA](https://www.jetbrains.com/idea/) +- Knowledge of Java, [Kotlin](https://kotlinlang.org), and [Dalvik bytecode](https://source.android.com/docs/core/runtime/dalvik-bytecode) +- Android reverse engineering skills and tools such as [jadx](https://github.com/skylot/jadx) + +## πŸƒ Prepare the environment + +Throughout the documentation, [ReVanced Patches](https://github.com/revanced/revanced-patches) will be used as an example project. + +1. Clone the repository + + ```bash + git clone https://github.com/revanced/revanced-patches && cd revanced-patches + ``` + +2. Build the project + + ```bash + ./gradlew build + ``` + + > [!NOTE] + > If the build fails due to authentication, you may need to authenticate to GitHub Packages. + > Create a PAT with the scope `read:packages` [here](https://github.com/settings/tokens/new?scopes=read:packages&description=ReVanced) and add your token to ~/.gradle/gradle.properties. + > + > Example `gradle.properties` file: + > + > ```properties + > gpr.user = user + > gpr.key = key + > ``` + +3. Open the project in your IDE + +> [!TIP] +> It is a good idea to set up a complete development environment for ReVanced, so that you can also test your patches by following the [ReVanced documentation](https://github.com/ReVanced/revanced-documentation). + +## ⏭️ What's next + +The next page will go into details about a ReVanced patch. + +Continue: [🧩 Anatomy of a patch](2_2_patch_anatomy.md) diff --git a/docs/2_2_1_fingerprinting.md b/docs/2_2_1_fingerprinting.md new file mode 100644 index 00000000..5ce35068 --- /dev/null +++ b/docs/2_2_1_fingerprinting.md @@ -0,0 +1,275 @@ +

+ + + + +
+ + + + + +     + + + + + +     + + + + + +     + + + + + +     + + + + + +     + + + + + +     + + + + + + +
+
+ Continuing the legacy of Vanced +

+ +# πŸ”Ž Fingerprinting + +In the context of ReVanced, fingerprinting is primarily used to resolve methods with a limited amount of known information. +Methods with obfuscated names that change with each update are primary candidates for fingerprinting. +The goal of fingerprinting is to uniquely identify a method by capturing various attributes, such as the return type, access flags, an opcode pattern, strings, and more. + +## ⛳️ Example fingerprint + +Throughout the documentation, the following example will be used to demonstrate the concepts of fingerprints: + +```kt + +package app.revanced.patches.ads.fingerprints + +object ShowAdsFingerprint : MethodFingerprint( + returnType = "Z", + accessFlags = AccessFlags.PUBLIC or AccessFlags.FINAL, + parameters = listOf("Z"), + opcodes = listOf(Opcode.RETURN), + strings = listOf("pro"), + customFingerprint = { (methodDef, classDef) -> methodDef.definingClass == "Lcom/some/app/ads/AdsLoader;" } +) +``` + +## πŸ”Ž Reconstructing the original code from a fingerprint + +The following code is reconstructed from the fingerprint to understand how a fingerprint is created. + +The fingerprint contains the following information: + +- Method signature: + + ```kt + returnType = "Z", + access = AccessFlags.PUBLIC or AccessFlags.FINAL, + parameters = listOf("Z"), + ``` + +- Method implementation: + + ```kt + opcodes = listOf(Opcode.RETURN) + strings = listOf("pro"), + ``` + +- Package and class name: + + ```kt + customFingerprint = { (methodDef, classDef) -> methodDef.definingClass == "Lcom/some/app/ads/AdsLoader;"} + ``` + +With this information, the original code can be reconstructed: + +```java + package com.some.app.ads; + + class AdsLoader { + public final boolean (boolean ) { + // ... + + var userStatus = "pro"; + + // ... + + return ; + } + } +``` + +> [!TIP] +> A fingerprint should contain information about a method likely to remain the same across updates. +> A method's name is not included in the fingerprint because it is likely to change with each update in an obfuscated app. In contrast, the return type, access flags, parameters, patterns of opcodes, and strings are likely to remain the same. + +## πŸ”¨ How to use fingerprints + +After creating a fingerprint, add it to the constructor of a `BytecodePatch`: + +```kt +object DisableAdsPatch : BytecodePatch( + setOf(ShowAdsFingerprint) +) { + // ... + } +``` + +> [!NOTE] +> Fingerprints passed to the constructor of `BytecodePatch` are resolved by ReVanced Patcher before the patch is executed. + +> [!TIP] +> Multiple patches can share fingerprints. If a fingerprint is resolved once, it will not be resolved again. + +> [!TIP] +> If a fingerprint has an opcode pattern, you can use the `FuzzyPatternScanMethod` annotation to fuzzy match the pattern. +> Opcode pattern arrays can contain `null` values to indicate that the opcode at the index is unknown. +> Any opcode will match to a `null` value. + +> [!WARNING] +> If the fingerprint can not be resolved because it does not match any method, the result of a fingerprint is `null`. + +Once the fingerprint is resolved, the result can be used in the patch: + +```kt +object DisableAdsPatch : BytecodePatch( + setOf(ShowAdsFingerprint) +) { + override fun execute(context: BytecodeContext) { + val result = ShowAdsFingerprint.result + ?: throw PatchException("ShowAdsFingerprint not found") + + // ... + } +} +``` + +The result of a fingerprint that resolved successfully contains mutable and immutable references to the method and the class it is defined in. + +```kt +class MethodFingerprintResult( + val method: Method, + val classDef: ClassDef, + val scanResult: MethodFingerprintScanResult, + // ... +) { + val mutableClass by lazy { /* ... */ } + val mutableMethod by lazy { /* ... */ } + + // ... +} + +class MethodFingerprintScanResult( + val patternScanResult: PatternScanResult?, + val stringsScanResult: StringsScanResult?, +) { + class StringsScanResult(val matches: List) { + class StringMatch(val string: String, val index: Int) + } + + class PatternScanResult( + val startIndex: Int, + val endIndex: Int, + // ... + ) { + // ... + } +} +``` + +## 🏹 Manual resolution of fingerprints + +Unless a fingerprint is added to the constructor of `BytecodePatch`, the fingerprint will not be resolved automatically by ReVanced Patcher before the patch is executed. +Instead, the fingerprint can be resolved manually using various overloads of the `resolve` function of a fingerprint. + +You can resolve a fingerprint in the following ways: + +- On a **list of classes**, if the fingerprint can resolve on a known subset of classes + + If you have a known list of classes you know the fingerprint can resolve on, you can resolve the fingerprint on the list of classes: + + ```kt + override fun execute(context: BytecodeContext) { + val result = ShowAdsFingerprint.also { it.resolve(context, context.classes) }.result + ?: throw PatchException("ShowAdsFingerprint not found") + + // ... + } + ``` + +- On a **single class**, if the fingerprint can resolve on a single known class + + If you know the fingerprint can resolve to a method in a specific class, you can resolve the fingerprint on the class: + + ```kt + override fun execute(context: BytecodeContext) { + val adsLoaderClass = context.classes.single { it.name == "Lcom/some/app/ads/Loader;" } + + val result = ShowAdsFingerprint.also { it.resolve(context, adsLoaderClass) }.result + ?: throw PatchException("ShowAdsFingerprint not found") + + // ... + } + ``` + +- On a **single method**, to extract certain information about a method + + The result of a fingerprint contains useful information about the method, such as the start and end index of an opcode pattern or the indices of the instructions with certain string references. + A fingerprint can be leveraged to extract such information from a method instead of manually figuring it out: + + ```kt + override fun execute(context: BytecodeContext) { + val adsFingerprintResult = ShowAdsFingerprint.result + ?: throw PatchException("ShowAdsFingerprint not found") + + val proStringsFingerprint = object : MethodFingerprint( + strings = listOf("free", "trial") + ) {} + + proStringsFingerprint.also { + it.resolve(context, adsFingerprintResult.method) + }.result?.let { result -> + result.scanResult.stringsScanResult!!.matches.forEach { match -> + println("The index of the string '${match.string}' is ${match.index}") + } + + } ?: throw PatchException("pro strings fingerprint not found") + } + ``` + +> [!TIP] +> To see real-world examples of fingerprints, check out the [ReVanced Patches](https://github.com/revanced/revanced-patches) repository. + +## ⏭️ What's next + +The next page discusses the structure and conventions of patches. + +Continue: [πŸ“œ Project structure and conventions](3_structure_and_conventions.md) diff --git a/docs/2_2_patch_anatomy.md b/docs/2_2_patch_anatomy.md new file mode 100644 index 00000000..9b941f3f --- /dev/null +++ b/docs/2_2_patch_anatomy.md @@ -0,0 +1,211 @@ +

+ + + + +
+ + + + + +     + + + + + +     + + + + + +     + + + + + +     + + + + + +     + + + + + +     + + + + + + +
+
+ Continuing the legacy of Vanced +

+ +# 🧩 Anatomy of a ReVanced patch + +Learn the API to create patches using ReVanced Patcher. + +## ⛳️ Example patch + +Throughout the documentation, the following example will be used to demonstrate the concepts of patches: + +```kt +package app.revanced.patches.ads + +@Patch( + name = "Disable ads", + description = "Disable ads in the app.", + dependencies = [DisableAdsResourcePatch::class], + compatiblePackages = [CompatiblePackage("com.some.app", ["1.3.0"])] +) +object DisableAdsPatch : BytecodePatch( + setOf(ShowAdsFingerprint) +) { + override fun execute(context: BytecodeContext) { + ShowAdsFingerprint.result?.let { result -> + result.mutableMethod.addInstructions( + 0, + """ + # Return false. + const/4 v0, 0x0 + return v0 + """ + ) + } ?: throw PatchException("ShowAdsFingerprint not found") + } +} +``` + +## πŸ”Ž Breakdown + +The example patch consists of the following parts: + +### πŸ“ Patch annotation + +```kt +@Patch( + name = "Disable ads", + description = "Disable ads in the app.", + dependencies = [DisableAdsResourcePatch::class], + compatiblePackages = [CompatiblePackage("com.some.app", ["1.3.0"])] +) +``` + +The `@Patch` annotation is used to provide metadata about the patch. + +Notable annotation parameters are: + +- `name`: The name of the patch. This is used as an identifier for the patch. + If this parameter is not set, `PatchBundleLoader` will not load the patch. + Other patches can still use this patch as a dependency +- `description`: A description of the patch. Can be unset if the name is descriptive enough +- `dependencies`: A set of patches which the patch depends on. The patches in this set will be executed before this patch. If a dependency patch raises an exception, this patch will not be executed; subsquently, other patches that depend on this patch will not be executed. +- `compatiblePackages`: A set of `CompatiblePackage` objects. Each `CompatiblePackage` object contains the package name and a set of compatible version names. This parameter can specify the packages and versions the patch is compatible with. Patches can still execute on incompatible packages, but it is recommended to use this parameter to list known compatible packages + - If unset, it is implied that the patch is compatible with all packages + - If the set of versions is unset, it is implied that the patch is compatible with all versions of the package + - If the set of versions is empty, it is implied that the patch is not compatible with any version of the package. This can be useful, for example, to prevent a patch from executing on specific packages that are known to be incompatible + +> [!WARNING] +> Circular dependencies are not allowed. If a patch depends on another patch, the other patch cannot depend on the first patch. + +> [!NOTE] +> The `@Patch` annotation is optional. If the patch does not require any metadata, it can be omitted. +> If the patch is only used as a dependency, the metadata, such as the `compatiblePackages` parameter, has no effect, as every dependency patch inherits the compatible packages of the patches that depend on it. + +> [!TIP] +> An abstract patch class can be annotated with `@Patch`. +> Patches extending off the abstract patch class will inherit the metadata of the abstract patch class. + +> [!TIP] +> Instead of the `@Patch` annotation, the superclass's constructor can be used. This is useful in the example scenario where you want to create an abstract patch class. +> +> Example: +> +> ```kt +> abstract class AbstractDisableAdsPatch( +> fingerprints: Set +> ) : BytecodePatch( +> name = "Disable ads", +> description = "Disable ads in the app.", +> fingerprints +> ) { +> // ... +> } +> ``` +> +> Remember that this constructor has precedence over the `@Patch` annotation. + +### πŸ—οΈ Patch class + +```kt +object DisableAdsPatch : BytecodePatch( /* Parameters */ ) { + // ... +} +``` + +Each patch class extends off a base class that implements the `Patch` interface. +The interface requires the `execute` method to be implemented. +Depending on which base class is extended, the patch can modify different parts of the APK as described in [🧩 Introduction to ReVanced Patches](2_introduction_to_patches.md). + +> [!TIP] +> A patch is usually a singleton object, meaning only one patch instance exists in the JVM. +> Because dependencies are executed before the patch itself, a patch can rely on the state of the dependency patch. +> This is useful in the example scenario, where the `DisableAdsPatch` depends on the `DisableAdsResourcePatch`. +> The `DisableAdsResourcePatch` can, for example, be used to read the decoded resources of the app and provide the `DisableAdsPatch` with the necessary information to disable ads because the `DisableAdsResourcePatch` is executed before the `DisableAdsPatch` and is a singleton object. + +### 🏁 The `execute` function + +The `execute` function is declared in the `Patch` interface and needs to be implemented. +The `execute` function receives an instance of a context object that provides access to the APK. The patch can use this context to modify the APK as described in [🧩 Introduction to ReVanced Patches](2_introduction_to_patches.md). + +In the current example, the patch adds instructions at the beginning of a method implementation in the Dalvik VM bytecode. The added instructions return `false` to disable ads in the current example: + +```kt +val result = LoadAdsFingerprint.result + ?: throw PatchException("LoadAdsFingerprint not found") + +result.mutableMethod.addInstructions( + 0, + """ + # Return false. + const/4 v0, 0x0 + return v0 + """ +) +``` + +> [!NOTE] +> This patch uses a fingerprint to find the method and replaces the method's instructions with new instructions. +> The fingerprint is resolved on the classes present in `BytecodeContext`. +> Fingerprints will be explained in more detail on the next page. + +> [!TIP] +> The patch can also raise any `Exception` or `Throwable` at any time to indicate that the patch failed to execute. A `PatchException` is recommended to be raised if the patch fails to execute. +> If any patch depends on this patch, the dependent patch will not be executed, whereas other patches that do not depend on this patch can still be executed. +> ReVanced Patcher will handle any exception raised by a patch. + +> [!TIP] +> To see real-world examples of patches, check out the [ReVanced Patches](https://github.com/revanced/revanced-patches) repository. + +## ⏭️ What's next + +The next page explains the concept of fingerprinting in ReVanced Patcher. + +Continue: [πŸ”Ž Fingerprinting](2_2_1_fingerprinting.md) diff --git a/docs/2_patches_intro.md b/docs/2_patches_intro.md new file mode 100644 index 00000000..bcb3ab22 --- /dev/null +++ b/docs/2_patches_intro.md @@ -0,0 +1,125 @@ +

+ + + + +
+ + + + + +     + + + + + +     + + + + + +     + + + + + +     + + + + + +     + + + + + +     + + + + + + +
+
+ Continuing the legacy of Vanced +

+ +# 🧩 Introduction to ReVanced Patches + +Learn the basic concepts of ReVanced Patcher and how to create patches. + +## πŸ“™ Fundamentals + +A patch is a piece of code that modifies an Android application. +There are multiple types of patches. Each type can modify a different part of the APK, such as the Dalvik VM bytecode, the APK resources, or arbitrary files in the APK: + +- A `BytecodePatch` modifies the Dalvik VM bytecode +- A `ResourcePatch` modifies (decoded) resources +- A `RawResourcePatch` modifies arbitrary files + +Each patch can declare a set of dependencies on other patches. ReVanced Patcher will first execute dependencies before executing the patch itself. This way, multiple patches can work together for abstract purposes in a modular way. + +A patch class can be annotated with `@Patch` to provide metadata about and dependencies of the patch. +Alternatively, a constructor of the superclass can be used. This is useful in the example scenario where you want to create an abstract patch class. + +The entry point of a patch is the `execute` function. This function is called by ReVanced Patcher when the patch is executed. The `execute` function receives an instance of the context object that provides access to the APK. The patch can use this context to modify the APK. + +Each type of context provides different APIs to modify the APK. For example, the `BytecodeContext` provides APIs to modify the Dalvik VM bytecode, while the `ResourceContext` provides APIs to modify resources. + +The difference between `ResourcePatch` and `RawResourcePatch` is that ReVanced Patcher will decode the resources if it is supplied a `ResourcePatch` for execution or if any kind of patch depends on a `ResourcePatch` and will not decode the resources before executing `RawResourcePatch`. Both, `ResourcePatch` and `RawResourcePatch` can modify arbitrary files in the APK, whereas only `ResourcePatch` can modify decoded resources. The choice of which type to use depends on the use case. Decoding and building resources is a time- and resource-consuming process, so if the patch does not need to modify decoded resources, it is better to use `RawResourcePatch` or `BytecodePatch`. + +Example of a `BytecodePatch`: + +```kt +@Surpress("unused") +object MyPatch : BytecodePatch() { + override fun execute(context: BytecodeContext) { + // Your patch code here + } +} +``` + +Example of a `ResourcePatch`: + +```kt +@Surpress("unused") +object MyPatch : ResourcePatch() { + override fun execute(context: ResourceContext) { + // Your patch code here + } +} +``` + +Example of a `RawResourcePatch`: + +```kt +@Surpress("unused") +object MyPatch : RawResourcePatch() { + override fun execute(context: ResourceContext) { + // Your patch code here + } +} +``` + +> [!TIP] +> To see real-world examples of patches, check out the [ReVanced Patches](https://github.com/revanced/revanced-patches) repository. + +## ⏭️ Whats next + +The next page will guide you through setting up a development environment for creating patches. + +Continue: [πŸ‘Ά Setting up a development environment](2_1_setup.md) diff --git a/docs/3_structure_and_conventions.md b/docs/3_structure_and_conventions.md new file mode 100644 index 00000000..03ad9fa9 --- /dev/null +++ b/docs/3_structure_and_conventions.md @@ -0,0 +1,97 @@ +

+ + + + +
+ + + + + +     + + + + + +     + + + + + +     + + + + + +     + + + + + +     + + + + + +     + + + + + + +
+
+ Continuing the legacy of Vanced +

+ +# πŸ“œ Project structure and conventions + +Over time, a specific project structure and conventions have been established. + +## πŸ“ File structure + +Patches are organized in a specific file structure. The file structure is as follows: + +```text +πŸ“¦your.patches.app.category + β”œ πŸ“‚fingerprints + β”œ β”œ πŸ”SomeFingerprintA.kt + β”œ β”” πŸ”SomeFingerprintB.kt + β”” 🧩SomePatch.kt +``` + +## πŸ“™ Conventions + +- πŸ”₯ Name a patch after what it does. For example, if a patch removes ads, name it `RemoveAdsPatch`. + If a patch changes the color of a button, name it `ChangeButtonColorPatch` +- πŸ”₯ Write the patch description in the third person, present tense, and end it with a period. + If a patch removes ads, the description can be omitted because of redundancy, but if a patch changes the color of a button, the description can be _Changes the color of the resume button to red._ +- πŸ”₯ Write patches with modularity and reusability in mind. Patches can depend on each other, so it is important to write patches in a way that can be used in different contexts. +- πŸ”₯πŸ”₯ Keep patches as minimal as possible. This reduces the risk of failing patches. + Instead of involving many abstract changes in one patch or writing entire methods or classes in a patch, + you can write code in integrations. Integrations are compiled classes that are merged into the app before patches are executed as described in [πŸ’‰ Introduction to ReVanced Patcher](1_patcher_intro). + Patches can then reference methods and classes from integrations. + A real-world example of integrations can be found in the [ReVanced Integrations](https://github.com/ReVanced/revanced-integrations) repository +- πŸ”₯πŸ”₯πŸ”₯ Do not overload a fingerprint with information about a method that's likely to change. + In the example of an obfuscated method, it's better to fingerprint the method by its return type and parameters rather than its name because the name is likely to change. An intelligent selection of an opcode pattern or strings in a method can result in a strong fingerprint dynamic to app updates. +- πŸ”₯πŸ”₯πŸ”₯ Document your patches. Patches are abstract by nature, so it is important to document parts of the code that are not self-explanatory. For example, explain why and how a certain method is patched or large blocks of instructions that are modified or added to a method + +## ⏭️ What's next + +The next page discusses useful APIs for patch development. + +Continue: [πŸ’ͺ Advanced APIs](4_apis.md) diff --git a/docs/4_apis.md b/docs/4_apis.md new file mode 100644 index 00000000..21c1cf7a --- /dev/null +++ b/docs/4_apis.md @@ -0,0 +1,23 @@ +# πŸ’ͺ Advanced APIs + +A handful of APIs are available to make patch development easier and more efficient. + +## πŸ“™ Overview + +1. πŸ‘Ή Create new mutable classes with `context.proxy(ClassDef)` +2. πŸ” Find and proxy existing classes with `BytecodeContext.findClass(Predicate)` +3. πŸƒβ€ Easily access referenced methods recursively by index with `BytecodeContext.toMethodWalker(Method)` +4. πŸ”¨ Make use of extension functions from `BytecodeUtils` and `ResourceUtils` with certain applications (Available in ReVanced Patches) +5. πŸ’Ύ Read and write (decoded) resources with `ResourceContext.get(Path, Boolean) ` +6. πŸ“ƒ Read and write DOM files using `ResourceContext.document` +7. πŸ”§ Equip patches with configurable options using `Patch.options` + +### 🧰 APIs + +> [!WARNING] +> This section is still under construction and may be incomplete. + +## πŸŽ‰ Afterword + +ReVanced Patcher is a powerful library to patch Android applications, offering a rich set of APIs to develop patches that outlive app updates. Patches make up ReVanced; without you, the community of patch developers, ReVanced would not be what it is today. We hope that this documentation has been helpful to you and are excited to see what you will create with ReVanced Patcher. If you have any questions or need help, talk to us on one of our platforms linked on [revanced.app](https://revanced.app) or open an issue in case of a bug or feature request, +ReVanced diff --git a/docs/README.md b/docs/README.md new file mode 100644 index 00000000..028389ff --- /dev/null +++ b/docs/README.md @@ -0,0 +1,73 @@ +

+ + + + +
+ + + + + +     + + + + + +     + + + + + +     + + + + + +     + + + + + +     + + + + + +     + + + + + + +
+
+ Continuing the legacy of Vanced +

+ +# πŸ’‰ Documentation of ReVanced Patcher + +This documentation contains the fundamentals of ReVanced Patcher and how to use ReVanced Patcher to create patches + +## πŸ“– Table of content + +1. [πŸ’‰ Introduction to ReVanced Patcher](1_patcher_intro.md) +2. [🧩 Introduction to ReVanced Patches](2_patches_intro.md) + 1. [πŸ‘Ά Setting up a development environment](2_1_setup.md) + 2. [🧩 Anatomy of a ReVanced patch](2_2_patch_anatomy.md) + 1. [πŸ”Ž Fingerprinting](2_2_1_fingerprinting.md) + 3. [πŸ“œ Project structure and conventions](3_structure_and_conventions.md) + 4. [πŸ’ͺ Advanced APIs](4_apis.md)