mirror of
https://github.com/NohamR/Tweaks.git
synced 2026-05-25 04:17:15 +00:00
Compare commits
12 Commits
ServerCatP
...
OqeePlus-i
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
4475f596d5 | ||
|
|
75f7add11f | ||
|
|
0fa7d613d8 | ||
|
|
432eafeb77 | ||
|
|
5a068bf71a | ||
|
|
9c7f571e8f | ||
|
|
6c5c90f060 | ||
|
|
d1af7d7a12 | ||
|
|
f69c20fb2c | ||
|
|
2e08be66c0 | ||
|
|
5973408630 | ||
|
|
a9ddc888f8 |
118
.github/workflows/build.yml
vendored
Normal file
118
.github/workflows/build.yml
vendored
Normal file
@@ -0,0 +1,118 @@
|
|||||||
|
name: Build Tweak
|
||||||
|
on:
|
||||||
|
workflow_dispatch:
|
||||||
|
inputs:
|
||||||
|
tweak:
|
||||||
|
description: 'Tweak folder to build (e.g., BusinessJB, CreditAgricoleJB, Infuse-iOS, TF1Plus/TF1Plus-tvOS)'
|
||||||
|
required: false
|
||||||
|
default: 'BusinessJB'
|
||||||
|
branch:
|
||||||
|
description: 'Branch to build from'
|
||||||
|
required: false
|
||||||
|
default: 'main'
|
||||||
|
|
||||||
|
concurrency:
|
||||||
|
group: ${{ github.workflow }}-${{ github.ref }}-${{ inputs.tweak || 'BusinessJB' }}
|
||||||
|
cancel-in-progress: true
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build:
|
||||||
|
name: Build ${{ inputs.tweak || 'BusinessJB' }}
|
||||||
|
runs-on: macos-latest
|
||||||
|
permissions:
|
||||||
|
contents: write
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout repository
|
||||||
|
uses: actions/checkout@v6
|
||||||
|
with:
|
||||||
|
ref: ${{ inputs.branch || github.ref }}
|
||||||
|
|
||||||
|
- name: Install dependencies
|
||||||
|
run: brew install make ldid
|
||||||
|
|
||||||
|
- name: Set PATH environment variables
|
||||||
|
run: |
|
||||||
|
echo "$(brew --prefix make)/libexec/gnubin" >> $GITHUB_PATH
|
||||||
|
echo "THEOS=${{ github.workspace }}/theos" >> $GITHUB_ENV
|
||||||
|
|
||||||
|
# Original from YTweaks
|
||||||
|
- name: Get Theos commit
|
||||||
|
run: |
|
||||||
|
get_commit_hash() {
|
||||||
|
local repo_url=$1
|
||||||
|
git ls-remote "$repo_url" HEAD | awk '{print substr($1,1,7)}'
|
||||||
|
}
|
||||||
|
TWEAK_PATH="${{ inputs.tweak || 'BusinessJB' }}"
|
||||||
|
if [[ "$TWEAK_PATH" == *"tvOS"* || "$TWEAK_PATH" == *"TVOS"* ]]; then
|
||||||
|
THEOS_URL="https://github.com/NohamR/theos-tvOS"
|
||||||
|
else
|
||||||
|
THEOS_URL="https://github.com/roothide/theos"
|
||||||
|
fi
|
||||||
|
echo "THEOS_REPO=$THEOS_URL" >> $GITHUB_ENV
|
||||||
|
echo "THEOS_COMMIT=$(get_commit_hash "$THEOS_URL")" >> $GITHUB_ENV
|
||||||
|
|
||||||
|
- name: Cache Theos
|
||||||
|
id: cache-theos
|
||||||
|
uses: actions/cache@v5
|
||||||
|
with:
|
||||||
|
path: theos
|
||||||
|
key: Tweak-18.6-SDK-${{ env.THEOS_COMMIT }}
|
||||||
|
|
||||||
|
- name: Setup Theos
|
||||||
|
if: ${{ steps.cache-theos.outputs.cache-hit != 'true' }}
|
||||||
|
run: |
|
||||||
|
git clone --quiet --depth=1 --recurse-submodules ${{ env.THEOS_REPO }}.git theos
|
||||||
|
git clone --quiet --depth=1 -n --filter=tree:0 https://github.com/Tonwalter888/iOS-SDKs.git
|
||||||
|
cd iOS-SDKs
|
||||||
|
git sparse-checkout set --no-cone iPhoneOS18.6.sdk
|
||||||
|
git checkout --quiet
|
||||||
|
mv *.sdk "$THEOS/sdks"
|
||||||
|
|
||||||
|
- name: Clone headers
|
||||||
|
run: |
|
||||||
|
if [ ! -d "$THEOS/include/PSHeader" ]; then
|
||||||
|
git clone --quiet --depth=1 https://github.com/PoomSmart/PSHeader.git "$THEOS/include/PSHeader"
|
||||||
|
else
|
||||||
|
cd $THEOS/include/PSHeader
|
||||||
|
git pull --quiet --force
|
||||||
|
cd ${{ github.workspace }}
|
||||||
|
fi
|
||||||
|
|
||||||
|
- name: Build Tweak
|
||||||
|
run: |
|
||||||
|
TWEAK_PATH="${{ inputs.tweak || 'BusinessJB' }}"
|
||||||
|
TWEAK_NAME=$(basename "$TWEAK_PATH")
|
||||||
|
|
||||||
|
cd "$TWEAK_PATH"
|
||||||
|
echo "TWEAK_VERSION=$(grep '^Version:' control | cut -d ' ' -f2)" >> $GITHUB_ENV
|
||||||
|
echo "TWEAK_NAME=$TWEAK_NAME" >> $GITHUB_ENV
|
||||||
|
|
||||||
|
# Check if this is a tvOS tweak
|
||||||
|
if grep -q "TARGET.*appletv" Makefile; then
|
||||||
|
# tvOS tweaks don't use rootless/roothide schemes
|
||||||
|
make clean package DEBUG=0 FINALPACKAGE=1
|
||||||
|
else
|
||||||
|
# iOS tweaks build with multiple schemes
|
||||||
|
make clean package DEBUG=0 FINALPACKAGE=1
|
||||||
|
make clean package DEBUG=0 FINALPACKAGE=1 THEOS_PACKAGE_SCHEME=rootless
|
||||||
|
make clean package DEBUG=0 FINALPACKAGE=1 THEOS_PACKAGE_SCHEME=roothide
|
||||||
|
fi
|
||||||
|
|
||||||
|
mv packages/*.deb ${{ github.workspace }}
|
||||||
|
|
||||||
|
- name: Upload the tweak .deb(s)
|
||||||
|
uses: actions/upload-artifact@v7
|
||||||
|
with:
|
||||||
|
name: ${{ env.TWEAK_NAME }} v${{ env.TWEAK_VERSION }}
|
||||||
|
path: ${{ github.workspace }}/*.deb
|
||||||
|
if-no-files-found: error
|
||||||
|
overwrite: true
|
||||||
|
|
||||||
|
- name: Create a draft release
|
||||||
|
uses: softprops/action-gh-release@v3
|
||||||
|
with:
|
||||||
|
tag_name: ${{ env.TWEAK_NAME }}-v${{ env.TWEAK_VERSION }}
|
||||||
|
name: ${{ env.TWEAK_NAME }} v${{ env.TWEAK_VERSION }}
|
||||||
|
files: ${{ github.workspace }}/*.deb
|
||||||
|
draft: true
|
||||||
3
.gitignore
vendored
3
.gitignore
vendored
@@ -1,5 +1,2 @@
|
|||||||
.DS_Store
|
.DS_Store
|
||||||
/CreditAgricoleTweak
|
|
||||||
/RMHook
|
|
||||||
/rootless
|
|
||||||
Build.md
|
Build.md
|
||||||
3
BusinessJB/.gitignore
vendored
Normal file
3
BusinessJB/.gitignore
vendored
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
.theos/
|
||||||
|
packages/
|
||||||
|
.DS_Store
|
||||||
7
BusinessJB/BusinessJB.plist
Normal file
7
BusinessJB/BusinessJB.plist
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
{
|
||||||
|
Filter = {
|
||||||
|
Bundles = (
|
||||||
|
"fr.labanquepostale.pmo.mescomptes",
|
||||||
|
);
|
||||||
|
};
|
||||||
|
}
|
||||||
13
BusinessJB/Makefile
Normal file
13
BusinessJB/Makefile
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
TARGET = iphone:latest:14.0
|
||||||
|
INSTALL_TARGET_PROCESSES = Business
|
||||||
|
ARCHS = arm64 arm64e
|
||||||
|
|
||||||
|
include $(THEOS)/makefiles/common.mk
|
||||||
|
|
||||||
|
TWEAK_NAME = BusinessJB
|
||||||
|
|
||||||
|
BusinessJB_FILES = Tweak.x
|
||||||
|
BusinessJB_CFLAGS = -fobjc-arc
|
||||||
|
BusinessJB_FRAMEWORKS = Foundation
|
||||||
|
|
||||||
|
include $(THEOS_MAKE_PATH)/tweak.mk
|
||||||
8
BusinessJB/Tweak.x
Normal file
8
BusinessJB/Tweak.x
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
#import <substrate.h>
|
||||||
|
#import <mach-o/dyld.h>
|
||||||
|
|
||||||
|
%hook JailbreakDetection
|
||||||
|
- (bool)jailbroken {
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
%end
|
||||||
9
BusinessJB/control
Normal file
9
BusinessJB/control
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
Package: xyz.nohamr.businessjb
|
||||||
|
Name: BusinessJB
|
||||||
|
Version: 1.0.0
|
||||||
|
Architecture: iphoneos-arm
|
||||||
|
Description: Bypass the jailbreak detection of the Business app.
|
||||||
|
Maintainer: NohamR
|
||||||
|
Author: NohamR
|
||||||
|
Section: Tweaks
|
||||||
|
Depends: mobilesubstrate (>= 0.9.5000)
|
||||||
26
BusinessJB/index.md
Normal file
26
BusinessJB/index.md
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
# BusinessJB
|
||||||
|
|
||||||
|
Disables jailbreak detection in La Banque Postale by forcing `JailbreakDetection` to return `FALSE`.
|
||||||
|
|
||||||
|
- **App**: [Business – La Banque Postale](https://apps.apple.com/fr/app/business-la-banque-postale/id1438089314)
|
||||||
|
- **Latest version**: 2.3.000
|
||||||
|
- **Tested on**: iOS 15.8.6
|
||||||
|
|
||||||
|
## Build
|
||||||
|
|
||||||
|
```sh
|
||||||
|
make clean && make package THEOS_PACKAGE_SCHEME=rootless DEBUG=0
|
||||||
|
```
|
||||||
|
|
||||||
|
## Inject
|
||||||
|
|
||||||
|
```sh
|
||||||
|
cyan -i com.labanquepostale.business-2.3.000.ipa \
|
||||||
|
-o com.labanquepostale.business-2.3.000_patched.ipa \
|
||||||
|
-f xyz.nohamr.businessjb_1.0.0-1_iphoneos-arm64.deb \
|
||||||
|
-u
|
||||||
|
```
|
||||||
|
|
||||||
|
## Screenshots
|
||||||
|
|
||||||
|

|
||||||
3
CreditAgricoleJB/.gitignore
vendored
Normal file
3
CreditAgricoleJB/.gitignore
vendored
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
.theos/
|
||||||
|
packages/
|
||||||
|
.DS_Store
|
||||||
7
CreditAgricoleJB/CreditAgricoleTweak.plist
Normal file
7
CreditAgricoleJB/CreditAgricoleTweak.plist
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
{
|
||||||
|
Filter = {
|
||||||
|
Bundles = (
|
||||||
|
"fr.creditagricole.monbudget",
|
||||||
|
);
|
||||||
|
};
|
||||||
|
}
|
||||||
13
CreditAgricoleJB/Makefile
Normal file
13
CreditAgricoleJB/Makefile
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
TARGET = iphone:latest:14.0
|
||||||
|
INSTALL_TARGET_PROCESSES = nmb_prod
|
||||||
|
ARCHS = arm64 arm64e
|
||||||
|
|
||||||
|
include $(THEOS)/makefiles/common.mk
|
||||||
|
|
||||||
|
TWEAK_NAME = CreditAgricoleTweak
|
||||||
|
|
||||||
|
CreditAgricoleTweak_FILES = Tweak.x
|
||||||
|
CreditAgricoleTweak_CFLAGS = -fobjc-arc
|
||||||
|
CreditAgricoleTweak_FRAMEWORKS = Foundation
|
||||||
|
|
||||||
|
include $(THEOS_MAKE_PATH)/tweak.mk
|
||||||
27
CreditAgricoleJB/Tweak.x
Normal file
27
CreditAgricoleJB/Tweak.x
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
#import <substrate.h>
|
||||||
|
#import <mach-o/dyld.h>
|
||||||
|
#import <string.h>
|
||||||
|
#import <Foundation/Foundation.h>
|
||||||
|
|
||||||
|
#define TARGET_MODULE "nmb_prod"
|
||||||
|
#define IDA_BASE 0x100000000
|
||||||
|
#define ADDR_JB_DETECT 0x10055F914 // Address of target function in IDA (adjust if needed)
|
||||||
|
|
||||||
|
static void *(*original_sub)(void);
|
||||||
|
|
||||||
|
static void *hooked_sub(void) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
%ctor {
|
||||||
|
for (uint32_t i = 0; i < _dyld_image_count(); i++) {
|
||||||
|
const char *name = _dyld_get_image_name(i);
|
||||||
|
if (name && strstr(name, TARGET_MODULE)) {
|
||||||
|
uintptr_t base = (uintptr_t)_dyld_get_image_header(i);
|
||||||
|
uintptr_t addr = base + (ADDR_JB_DETECT - IDA_BASE);
|
||||||
|
MSHookFunction((void *)addr, (void *)hooked_sub, (void **)&original_sub);
|
||||||
|
NSLog(@"[CreditAgricoleTweak] Hooked function at 0x%lx", addr);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
9
CreditAgricoleJB/control
Normal file
9
CreditAgricoleJB/control
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
Package: xyz.nohamr.creditagricolejb
|
||||||
|
Name: CreditAgricoleJB (Rootless)
|
||||||
|
Version: 1.0.0
|
||||||
|
Architecture: iphoneos-arm
|
||||||
|
Description: Bypas jailbreak detection in Credit Agricole app
|
||||||
|
Maintainer: NohamR
|
||||||
|
Author: NohamR
|
||||||
|
Section: Tweaks
|
||||||
|
Depends: mobilesubstrate (>= 0.9.5000)
|
||||||
26
CreditAgricoleJB/index.md
Normal file
26
CreditAgricoleJB/index.md
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
# CreditAgricoleJB
|
||||||
|
|
||||||
|
Disables jailbreak detection in Ma Banque.
|
||||||
|
|
||||||
|
- **App**: [Ma Banque](https://apps.apple.com/fr/app/ma-banque/id376202925)
|
||||||
|
- **Latest version**: 47.0.0
|
||||||
|
- **Tested on**: iOS 15.8.6
|
||||||
|
|
||||||
|
## Build
|
||||||
|
|
||||||
|
```sh
|
||||||
|
make clean && make package THEOS_PACKAGE_SCHEME=rootless DEBUG=0
|
||||||
|
```
|
||||||
|
|
||||||
|
## Inject
|
||||||
|
|
||||||
|
```sh
|
||||||
|
cyan -i com.creditagricole.mabanque-47.0.0.ipa \
|
||||||
|
-o com.creditagricole.mabanque-47.0.0_patched.ipa \
|
||||||
|
-f xyz.nohamr.creditagricolejb_1.0.0-1_iphoneos-arm64.deb \
|
||||||
|
-u
|
||||||
|
```
|
||||||
|
|
||||||
|
## Screenshots
|
||||||
|
|
||||||
|

|
||||||
3
Infuse/Infuse-iOS/.gitignore
vendored
Normal file
3
Infuse/Infuse-iOS/.gitignore
vendored
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
.theos/
|
||||||
|
packages/
|
||||||
|
.DS_Store
|
||||||
7
Infuse/Infuse-iOS/Infuse.plist
Normal file
7
Infuse/Infuse-iOS/Infuse.plist
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
{
|
||||||
|
Filter = {
|
||||||
|
Bundles = (
|
||||||
|
"com.firecore.infuse",
|
||||||
|
);
|
||||||
|
};
|
||||||
|
}
|
||||||
13
Infuse/Infuse-iOS/Makefile
Normal file
13
Infuse/Infuse-iOS/Makefile
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
TARGET = iphone:latest:14.0
|
||||||
|
INSTALL_TARGET_PROCESSES = infuse
|
||||||
|
ARCHS = arm64 arm64e
|
||||||
|
|
||||||
|
include $(THEOS)/makefiles/common.mk
|
||||||
|
|
||||||
|
TWEAK_NAME = Infuse
|
||||||
|
|
||||||
|
Infuse_FILES = Tweak.x
|
||||||
|
Infuse_CFLAGS = -fobjc-arc
|
||||||
|
Infuse_FRAMEWORKS = Foundation
|
||||||
|
|
||||||
|
include $(THEOS_MAKE_PATH)/tweak.mk
|
||||||
81
Infuse/Infuse-iOS/Tweak.x
Normal file
81
Infuse/Infuse-iOS/Tweak.x
Normal file
@@ -0,0 +1,81 @@
|
|||||||
|
// changes : FCInAppPurchaseServiceFreemium block deleted, FCTraktIAPManager block deleted, layoutSubviews -> awakeFromNib on FCVersionView, and the dead return %orig lines after return statements cleaned up (they were unreachable in the old version too)
|
||||||
|
|
||||||
|
#import <substrate.h>
|
||||||
|
#import <Foundation/Foundation.h>
|
||||||
|
#import <UIKit/UIKit.h>
|
||||||
|
|
||||||
|
%hook FCIAPGUIHelper
|
||||||
|
+(bool) isProAvailable {
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
+(bool) isSubscriptionBought {
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
%end
|
||||||
|
|
||||||
|
%hook FCInAppPurchaseServiceBase
|
||||||
|
- (bool)isFeaturePurchased:(long long)arg1 tillDate:(id*)arg2 {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
- (bool)isFeaturePurchased:(long long)arg1 {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
%end
|
||||||
|
|
||||||
|
%hook FCInAppPurchaseServiceDummy
|
||||||
|
- (bool)isFeaturePurchased:(long long)arg1 tillDate:(id*)arg2 {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
%end
|
||||||
|
|
||||||
|
// REMOVED: FCInAppPurchaseServiceFreemium (class gone from binary, replaced by SK2)
|
||||||
|
|
||||||
|
// SK2 IAP backend
|
||||||
|
%hook _TtC6infuse31InAppPurchaseServiceFreemiumSK2
|
||||||
|
- (bool)isFeaturePurchased:(long long)arg1 tillDate:(id*)arg2 {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
- (long long)iapVersionStatus {
|
||||||
|
// FCUpgradeToProViewController.featureHasBought checks iapVersionStatus > 0
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
%end
|
||||||
|
|
||||||
|
%hook FCProductCollectionCell
|
||||||
|
-(bool) featurePurchased {
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
%end
|
||||||
|
|
||||||
|
// REMOVED: FCTraktIAPManager (class gone from binary)
|
||||||
|
|
||||||
|
%hook FCUpgradeToProViewController
|
||||||
|
-(bool) featureHasBought {
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
%end
|
||||||
|
|
||||||
|
// Add credits
|
||||||
|
@interface FCVersionView : UIView
|
||||||
|
@property (nonatomic, strong) UILabel *label;
|
||||||
|
@end
|
||||||
|
|
||||||
|
%hook FCVersionView
|
||||||
|
- (void)awakeFromNib {
|
||||||
|
%orig;
|
||||||
|
UILabel *label = (UILabel *)[self valueForKey:@"label"];
|
||||||
|
if ([label.text containsString:@"Infuse Pro"] && ![label.text hasPrefix:@"Infuse Team •"]) {
|
||||||
|
label.text = [NSString stringWithFormat:@"Infuse Team • %@", label.text];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
%end
|
||||||
|
|
||||||
|
// iOS 16 Crash Fix
|
||||||
|
%hook CKContainer
|
||||||
|
+ (id)defaultContainer {
|
||||||
|
return nil;
|
||||||
|
}
|
||||||
|
+ (id)containerWithIdentifier:(id)arg1 {
|
||||||
|
return nil;
|
||||||
|
}
|
||||||
|
%end
|
||||||
10
Infuse/Infuse-iOS/control
Normal file
10
Infuse/Infuse-iOS/control
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
Package: io.infuseteam.infuserootless
|
||||||
|
Name: Infuse (Rootless)
|
||||||
|
Version: 2.0
|
||||||
|
Architecture: iphoneos-arm
|
||||||
|
Description: Unlock the full potential of Infuse
|
||||||
|
Maintainer: Infuse Team
|
||||||
|
Author: Infuse Team
|
||||||
|
Section: Tweaks
|
||||||
|
Depends: firmware (>= 18.3)
|
||||||
|
Conflicts: io.infuseteam.infuse, io.infuseteam.infuseroothide, io.infuseteam.infuselegacy, xyz.eshaydev.ignition, com.ippteam.infuse, com.strejda603.infuse6pro, xyz.eshaydev.infuse
|
||||||
26
Infuse/Infuse-iOS/index.md
Normal file
26
Infuse/Infuse-iOS/index.md
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
# Infuse (iOS)
|
||||||
|
|
||||||
|
Unlock the full potential of Infuse.
|
||||||
|
|
||||||
|
- **App**: [Infuse](https://apps.apple.com/fr/app/infuse/id1136220934)
|
||||||
|
- **Tested version**: 8.4.2
|
||||||
|
- **Target**: iOS 18+ (tested on iOS 18.7.1 using LiveContainer)
|
||||||
|
|
||||||
|
## Build
|
||||||
|
|
||||||
|
```sh
|
||||||
|
make package FINALPACKAGE=1
|
||||||
|
```
|
||||||
|
|
||||||
|
## Inject
|
||||||
|
|
||||||
|
```sh
|
||||||
|
cyan -i infuse-8.4.2.ipa \
|
||||||
|
-o infuse-8.4.2_patched.ipa \
|
||||||
|
-f io.infuseteam.infuserootless_2.0_iphoneos-arm.deb \
|
||||||
|
-u
|
||||||
|
```
|
||||||
|
|
||||||
|
## Screenshots
|
||||||
|
|
||||||
|

|
||||||
3
Infuse/Infuse-tvOS/.gitignore
vendored
Normal file
3
Infuse/Infuse-tvOS/.gitignore
vendored
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
.theos/
|
||||||
|
packages/
|
||||||
|
.DS_Store
|
||||||
7
Infuse/Infuse-tvOS/Infuse.plist
Normal file
7
Infuse/Infuse-tvOS/Infuse.plist
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
{
|
||||||
|
Filter = {
|
||||||
|
Bundles = (
|
||||||
|
"com.firecore.infuse",
|
||||||
|
);
|
||||||
|
};
|
||||||
|
}
|
||||||
17
Infuse/Infuse-tvOS/Makefile
Normal file
17
Infuse/Infuse-tvOS/Makefile
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
TARGET = appletv:latest:18.3
|
||||||
|
ARCHS = arm64
|
||||||
|
INSTALL_TARGET_PROCESSES = infuse
|
||||||
|
THEOS_PACKAGE_SCHEME = rootless
|
||||||
|
|
||||||
|
include $(THEOS)/makefiles/common.mk
|
||||||
|
|
||||||
|
TWEAK_NAME = Infuse
|
||||||
|
|
||||||
|
Infuse_FILES = Tweak.x
|
||||||
|
Infuse_CFLAGS = -fobjc-arc
|
||||||
|
Infuse_FRAMEWORKS = Foundation UIKit
|
||||||
|
|
||||||
|
include $(THEOS_MAKE_PATH)/tweak.mk
|
||||||
|
test-print:
|
||||||
|
@echo "ARCH is $(THEOS_PACKAGE_ARCH)"
|
||||||
|
@echo "CONTROL is $(_THEOS_DEB_PACKAGE_CONTROL_PATH)"
|
||||||
81
Infuse/Infuse-tvOS/Tweak.x
Normal file
81
Infuse/Infuse-tvOS/Tweak.x
Normal file
@@ -0,0 +1,81 @@
|
|||||||
|
// changes : FCInAppPurchaseServiceFreemium block deleted, FCTraktIAPManager block deleted, layoutSubviews -> awakeFromNib on FCVersionView, and the dead return %orig lines after return statements cleaned up (they were unreachable in the old version too)
|
||||||
|
|
||||||
|
#import <substrate.h>
|
||||||
|
#import <Foundation/Foundation.h>
|
||||||
|
#import <UIKit/UIKit.h>
|
||||||
|
|
||||||
|
%hook FCIAPGUIHelper
|
||||||
|
+(bool) isProAvailable {
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
+(bool) isSubscriptionBought {
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
%end
|
||||||
|
|
||||||
|
%hook FCInAppPurchaseServiceBase
|
||||||
|
- (bool)isFeaturePurchased:(long long)arg1 tillDate:(id*)arg2 {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
- (bool)isFeaturePurchased:(long long)arg1 {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
%end
|
||||||
|
|
||||||
|
%hook FCInAppPurchaseServiceDummy
|
||||||
|
- (bool)isFeaturePurchased:(long long)arg1 tillDate:(id*)arg2 {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
%end
|
||||||
|
|
||||||
|
// REMOVED: FCInAppPurchaseServiceFreemium (class gone from binary, replaced by SK2)
|
||||||
|
|
||||||
|
// SK2 IAP backend
|
||||||
|
%hook _TtC6infuse31InAppPurchaseServiceFreemiumSK2
|
||||||
|
- (bool)isFeaturePurchased:(long long)arg1 tillDate:(id*)arg2 {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
- (long long)iapVersionStatus {
|
||||||
|
// FCUpgradeToProViewController.featureHasBought checks iapVersionStatus > 0
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
%end
|
||||||
|
|
||||||
|
%hook FCProductCollectionCell
|
||||||
|
-(bool) featurePurchased {
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
%end
|
||||||
|
|
||||||
|
// REMOVED: FCTraktIAPManager (class gone from binary)
|
||||||
|
|
||||||
|
%hook FCUpgradeToProViewController
|
||||||
|
-(bool) featureHasBought {
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
%end
|
||||||
|
|
||||||
|
// Add credits
|
||||||
|
@interface FCVersionView : UIView
|
||||||
|
@property (nonatomic, strong) UILabel *label;
|
||||||
|
@end
|
||||||
|
|
||||||
|
%hook FCVersionView
|
||||||
|
- (void)awakeFromNib {
|
||||||
|
%orig;
|
||||||
|
UILabel *label = (UILabel *)[self valueForKey:@"label"];
|
||||||
|
if ([label.text containsString:@"Infuse Pro"] && ![label.text hasPrefix:@"Infuse Team •"]) {
|
||||||
|
label.text = [NSString stringWithFormat:@"Infuse Team • %@", label.text];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
%end
|
||||||
|
|
||||||
|
// iOS 16 Crash Fix
|
||||||
|
%hook CKContainer
|
||||||
|
+ (id)defaultContainer {
|
||||||
|
return nil;
|
||||||
|
}
|
||||||
|
+ (id)containerWithIdentifier:(id)arg1 {
|
||||||
|
return nil;
|
||||||
|
}
|
||||||
|
%end
|
||||||
10
Infuse/Infuse-tvOS/control
Normal file
10
Infuse/Infuse-tvOS/control
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
Package: io.infuseteam.infuserootless
|
||||||
|
Name: Infuse (Rootless)
|
||||||
|
Version: 2.0
|
||||||
|
Architecture: appletvos-arm64
|
||||||
|
Description: Unlock the full potential of Infuse
|
||||||
|
Maintainer: Infuse Team
|
||||||
|
Author: Infuse Team
|
||||||
|
Section: Tweaks
|
||||||
|
Depends: firmware
|
||||||
|
Conflicts: io.infuseteam.infuse, io.infuseteam.infuseroothide, io.infuseteam.infuselegacy, xyz.eshaydev.ignition, com.ippteam.infuse, com.strejda603.infuse6pro, xyz.eshaydev.infuse
|
||||||
26
Infuse/Infuse-tvOS/index.md
Normal file
26
Infuse/Infuse-tvOS/index.md
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
# Infuse tvOS
|
||||||
|
|
||||||
|
Unlock pro features in Infuse.
|
||||||
|
|
||||||
|
- **App**: [Infuse](https://apps.apple.com/fr/app/infuse/id1136220934)
|
||||||
|
- **Tested version**: 8.2.4
|
||||||
|
- **Target**: tvOS 18.3 (tested on tvOS 18.3)
|
||||||
|
|
||||||
|
## Build
|
||||||
|
|
||||||
|
```sh
|
||||||
|
make package FINALPACKAGE=1
|
||||||
|
```
|
||||||
|
|
||||||
|
## Inject
|
||||||
|
|
||||||
|
```sh
|
||||||
|
cyan -i infuse-8.2.4.ipa \
|
||||||
|
-o infuse-8.2.4_patched.ipa \
|
||||||
|
-f io.infuseteam.infuserootless_2.0_tvos-arm64.deb \
|
||||||
|
-u
|
||||||
|
```
|
||||||
|
|
||||||
|
## Screenshots
|
||||||
|
|
||||||
|

|
||||||
3
OqeePlus/OqeePlus-iOS/.gitignore
vendored
Normal file
3
OqeePlus/OqeePlus-iOS/.gitignore
vendored
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
.theos/
|
||||||
|
packages/
|
||||||
|
.DS_Store
|
||||||
12
OqeePlus/OqeePlus-iOS/Makefile
Normal file
12
OqeePlus/OqeePlus-iOS/Makefile
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
TARGET = iphone:latest:14.0
|
||||||
|
ARCHS = arm64
|
||||||
|
INSTALL_TARGET_PROCESSES = Oqee
|
||||||
|
|
||||||
|
include $(THEOS)/makefiles/common.mk
|
||||||
|
|
||||||
|
TWEAK_NAME = OqeePlus
|
||||||
|
|
||||||
|
OqeePlus_FILES = Tweak.x
|
||||||
|
OqeePlus_CFLAGS = -fobjc-arc
|
||||||
|
|
||||||
|
include $(THEOS_MAKE_PATH)/tweak.mk
|
||||||
10
OqeePlus/OqeePlus-iOS/OqeePlus.plist
Normal file
10
OqeePlus/OqeePlus-iOS/OqeePlus.plist
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
{
|
||||||
|
Filter = {
|
||||||
|
Bundles = (
|
||||||
|
"net.oqee.appleos",
|
||||||
|
);
|
||||||
|
Executables = (
|
||||||
|
App,
|
||||||
|
);
|
||||||
|
};
|
||||||
|
}
|
||||||
143
OqeePlus/OqeePlus-iOS/Tweak.x
Normal file
143
OqeePlus/OqeePlus-iOS/Tweak.x
Normal file
@@ -0,0 +1,143 @@
|
|||||||
|
#import <substrate.h>
|
||||||
|
#import <UIKit/UIKit.h>
|
||||||
|
#import <AVFoundation/AVFoundation.h>
|
||||||
|
#import <Foundation/Foundation.h>
|
||||||
|
#include <mach-o/dyld.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <stdint.h>
|
||||||
|
|
||||||
|
#define TAG @"[OQAdsLogger]"
|
||||||
|
|
||||||
|
%hook IMAAdsRequest
|
||||||
|
- (instancetype)initWithAdsResponse:(NSString *)adsResponse
|
||||||
|
adDisplayContainer:(id)adDisplayContainer
|
||||||
|
avPlayerVideoDisplay:(id)avPlayerVideoDisplay
|
||||||
|
pictureInPictureProxy:(id)pipProxy
|
||||||
|
userContext:(id)userContext {
|
||||||
|
|
||||||
|
NSLog(@"%@-[IMAAdsRequest init] [PiP path]", TAG);
|
||||||
|
NSLog(@"%@ adDisplayContainer = %@", TAG, adDisplayContainer);
|
||||||
|
NSLog(@"%@ avPlayerVideoDisplay = %@", TAG, avPlayerVideoDisplay);
|
||||||
|
NSLog(@"%@ pictureInPictureProxy = %@", TAG, pipProxy);
|
||||||
|
NSLog(@"%@ VMAP payload (%lu bytes):\n%@",
|
||||||
|
TAG, (unsigned long)adsResponse.length, adsResponse);
|
||||||
|
|
||||||
|
id result = %orig;
|
||||||
|
NSLog(@"%@ -> IMAAdsRequest = %p", TAG, result);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
// - (instancetype)initWithAdsResponse:(NSString *)adsResponse
|
||||||
|
// adDisplayContainer:(id)adDisplayContainer
|
||||||
|
// contentPlayhead:(id)contentPlayhead
|
||||||
|
// userContext:(id)userContext {
|
||||||
|
|
||||||
|
// NSLog(@"%@-[IMAAdsRequest init] [non-PiP path]", TAG);
|
||||||
|
// NSLog(@"%@ adDisplayContainer = %@", TAG, adDisplayContainer);
|
||||||
|
// NSLog(@"%@ contentPlayhead = %@", TAG, contentPlayhead);
|
||||||
|
// NSLog(@"%@ VMAP payload (%lu bytes):\n%@",
|
||||||
|
// TAG, (unsigned long)adsResponse.length, adsResponse);
|
||||||
|
|
||||||
|
// id result = %orig;
|
||||||
|
// NSLog(@"%@ -> IMAAdsRequest = %p", TAG, result);
|
||||||
|
// return result;
|
||||||
|
// }
|
||||||
|
|
||||||
|
// - (instancetype)initWithAdsResponse:(NSString *)adsResponse
|
||||||
|
// adDisplayContainer:(id)adDisplayContainer
|
||||||
|
// contentPlayhead:(id)contentPlayhead
|
||||||
|
// userContext:(id)userContext {
|
||||||
|
// NSString *tweakedVMAP = @"<?xml version='1.0' encoding='utf-8'?>\n"
|
||||||
|
// @"<vmap:VMAP\n"
|
||||||
|
// @" xmlns:vmap=\"http://www.iab.net/vmap-1.0\" version=\"1.0\">\n"
|
||||||
|
// @" <vmap:AdBreak breakId=\"pre-roll-intro\" breakType=\"linear\" timeOffset=\"start\">\n"
|
||||||
|
// @" <vmap:AdSource allowMultipleAds=\"false\" followRedirects=\"true\" id=\"1\">\n"
|
||||||
|
// @" <vmap:VASTAdData>\n"
|
||||||
|
// @" <VAST\n"
|
||||||
|
// @" xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\" version=\"3.0\" xsi:noNamespaceSchemaLocation=\"vast.xsd\">\n"
|
||||||
|
// @" <Ad id=\"breakIntro\" sequence=\"1\">\n"
|
||||||
|
// @" <InLine>\n"
|
||||||
|
// @" <AdSystem>VizChoice</AdSystem>\n"
|
||||||
|
// @" <AdTitle>Oqee Cine Break Intro</AdTitle>\n"
|
||||||
|
// @" <Creatives>\n"
|
||||||
|
// @" <Creative>\n"
|
||||||
|
// @" <Linear>\n"
|
||||||
|
// @" <Duration>00:00:03</Duration>\n"
|
||||||
|
// @" <MediaFiles>\n"
|
||||||
|
// @" <MediaFile delivery=\"progressive\" width=\"1920\" height=\"1080\" type=\"video/mp4\">https://noh.am/rick.mp4</MediaFile>\n"
|
||||||
|
// @" <MediaFile delivery=\"streaming\" width=\"1920\" height=\"1080\" type=\"application/dash+xml\">https://replay-01.bzn.oqee.net/oqee-static/barkers/oqee-cine/2025-10-21/adaptative/playlist_214c3e5132137e02.mpd</MediaFile>\n"
|
||||||
|
// @" <MediaFile delivery=\"streaming\" width=\"1920\" height=\"1080\" type=\"application/x-mpegURL\">https://replay-01.bzn.oqee.net/oqee-static/barkers/oqee-cine/2025-10-21/adaptative/playlist_214c3e5132137e02.m3u8</MediaFile>\n"
|
||||||
|
// @" </MediaFiles>\n"
|
||||||
|
// @" </Linear>\n"
|
||||||
|
// @" </Creative>\n"
|
||||||
|
// @" </Creatives>\n"
|
||||||
|
// @" </InLine>\n"
|
||||||
|
// @" </Ad>\n"
|
||||||
|
// @" </VAST>\n"
|
||||||
|
// @" </vmap:VASTAdData>\n"
|
||||||
|
// @" </vmap:AdSource>\n"
|
||||||
|
// @" </vmap:AdBreak>\n"
|
||||||
|
// @"</vmap:VMAP>\n";
|
||||||
|
// NSLog(@"%@ Replaced VMAP (%lu bytes) with tweaked document", TAG, adsResponse.length);
|
||||||
|
// return %orig(tweakedVMAP, adDisplayContainer, contentPlayhead, userContext);
|
||||||
|
// }
|
||||||
|
|
||||||
|
- (instancetype)initWithAdsResponse:(NSString *)adsResponse
|
||||||
|
adDisplayContainer:(id)adDisplayContainer
|
||||||
|
contentPlayhead:(id)contentPlayhead
|
||||||
|
userContext:(id)userContext {
|
||||||
|
// Replace the VMAP with an empty document
|
||||||
|
NSString *emptyVMAP = @"<?xml version='1.0' encoding='utf-8'?>"
|
||||||
|
@"<vmap:VMAP xmlns:vmap=\"http://www.iab.net/vmap-1.0\" version=\"1.0\"/>";
|
||||||
|
NSLog(@"%@ Replaced VMAP (%lu bytes) with empty document", TAG, adsResponse.length);
|
||||||
|
return %orig(emptyVMAP, adDisplayContainer, contentPlayhead, userContext);
|
||||||
|
}
|
||||||
|
%end
|
||||||
|
|
||||||
|
// IMAAdsLoader — confirms dispatch to IMA SDK
|
||||||
|
//
|
||||||
|
// Called from OQPlayerAdsLoader.requestAds(_:loader:) (sub_10008FC08)
|
||||||
|
// after Swift dynamic-cast guards pass. This is the point of no return —
|
||||||
|
// the IMA SDK takes ownership of the request and starts network I/O.
|
||||||
|
|
||||||
|
%hook IMAAdsLoader
|
||||||
|
- (void)requestAdsWithRequest:(id)request {
|
||||||
|
NSLog(@"%@ -[IMAAdsLoader requestAdsWithRequest:]", TAG);
|
||||||
|
NSLog(@"%@ loader = %@", TAG, self);
|
||||||
|
NSLog(@"%@ request = %@", TAG, [request debugDescription]);
|
||||||
|
%orig;
|
||||||
|
NSLog(@"%@ requestAdsWithRequest dispatched ✓", TAG);
|
||||||
|
}
|
||||||
|
%end
|
||||||
|
|
||||||
|
// IMAAdsManager — SDK parsed the VMAP, breaks are scheduled
|
||||||
|
//
|
||||||
|
// Called from the IMAAdsLoaderDelegate callback in OQImaManager
|
||||||
|
// At this point the IMA SDK has parsed the VMAP and knows all ad break
|
||||||
|
// positions. Passing nil for renderingSettings means OQEE uses defaults.
|
||||||
|
|
||||||
|
%hook IMAAdsManager
|
||||||
|
- (void)initializeWithAdsRenderingSettings:(id)renderingSettings {
|
||||||
|
NSLog(@"%@ -[IMAAdsManager initializeWithAdsRenderingSettings:]", TAG);
|
||||||
|
NSLog(@"%@ adsManager = %@", TAG, self);
|
||||||
|
NSLog(@"%@ renderingSettings = %@", TAG, renderingSettings ?: @"(nil — default)");
|
||||||
|
%orig;
|
||||||
|
NSLog(@"%@ VMAP parsed, ad breaks scheduled ✓", TAG);
|
||||||
|
}
|
||||||
|
%end
|
||||||
|
|
||||||
|
|
||||||
|
%hook VSSubscriptionRegistrationCenter
|
||||||
|
- (void)setCurrentSubscription:(id)subscription
|
||||||
|
{
|
||||||
|
NSLog(@"Blocked VSSubscriptionRegistrationCenter");
|
||||||
|
NSLog(@"Subscription: %@", subscription);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
%end
|
||||||
|
|
||||||
|
%ctor {
|
||||||
|
// activate dev mode
|
||||||
|
NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
|
||||||
|
[defaults setBool:YES forKey:@"tv.oqee.devModeEnabled"];
|
||||||
|
}
|
||||||
9
OqeePlus/OqeePlus-iOS/control
Normal file
9
OqeePlus/OqeePlus-iOS/control
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
Package: xyz.nohamr.oqeeplus
|
||||||
|
Name: Oqee+ (iOS)
|
||||||
|
Version: 1.0
|
||||||
|
Architecture: iphoneos-arm64
|
||||||
|
Description: Oqee+ Ads blocker hook for iOS
|
||||||
|
Maintainer: NohamR
|
||||||
|
Author: NohamR
|
||||||
|
Section: Tweaks
|
||||||
|
Depends: mobilesubstrate (>= 0.9.5000)
|
||||||
22
OqeePlus/OqeePlus-iOS/index.md
Normal file
22
OqeePlus/OqeePlus-iOS/index.md
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
# OqeePlus iOS
|
||||||
|
|
||||||
|
Block ads initialization on Oqee.
|
||||||
|
|
||||||
|
- **App**: [OqeePlus : Streaming, TV en Direct](https://apps.apple.com/fr/app/free-tv/id1542614107)
|
||||||
|
- **Tested version**: 2.40
|
||||||
|
- **Target**: iOS
|
||||||
|
|
||||||
|
## Build
|
||||||
|
|
||||||
|
```sh
|
||||||
|
make package FINALPACKAGE=1
|
||||||
|
```
|
||||||
|
|
||||||
|
## Inject
|
||||||
|
|
||||||
|
```sh
|
||||||
|
cyan -i oqeeplus.ipa \
|
||||||
|
-o oqeeplus_patched.ipa \
|
||||||
|
-f xyz.nohamr.oqeeplus_1.0_iphoneos-arm64.deb \
|
||||||
|
-u
|
||||||
|
```
|
||||||
3
OqeePlus/OqeePlus-tvOS/.gitignore
vendored
Normal file
3
OqeePlus/OqeePlus-tvOS/.gitignore
vendored
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
.theos/
|
||||||
|
packages/
|
||||||
|
.DS_Store
|
||||||
13
OqeePlus/OqeePlus-tvOS/Makefile
Normal file
13
OqeePlus/OqeePlus-tvOS/Makefile
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
TARGET = appletv:latest:18.3
|
||||||
|
ARCHS = arm64
|
||||||
|
INSTALL_TARGET_PROCESSES = Oqee
|
||||||
|
THEOS_PACKAGE_SCHEME = rootless
|
||||||
|
|
||||||
|
include $(THEOS)/makefiles/common.mk
|
||||||
|
|
||||||
|
TWEAK_NAME = OqeePlus
|
||||||
|
|
||||||
|
OqeePlus_FILES = Tweak.x
|
||||||
|
OqeePlus_CFLAGS = -fobjc-arc
|
||||||
|
|
||||||
|
include $(THEOS_MAKE_PATH)/tweak.mk
|
||||||
10
OqeePlus/OqeePlus-tvOS/OqeePlus.plist
Normal file
10
OqeePlus/OqeePlus-tvOS/OqeePlus.plist
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
{
|
||||||
|
Filter = {
|
||||||
|
Bundles = (
|
||||||
|
"net.oqee.appleos",
|
||||||
|
);
|
||||||
|
Executables = (
|
||||||
|
App,
|
||||||
|
);
|
||||||
|
};
|
||||||
|
}
|
||||||
88
OqeePlus/OqeePlus-tvOS/Tweak.x
Normal file
88
OqeePlus/OqeePlus-tvOS/Tweak.x
Normal file
@@ -0,0 +1,88 @@
|
|||||||
|
#import <substrate.h>
|
||||||
|
#import <UIKit/UIKit.h>
|
||||||
|
#import <AVFoundation/AVFoundation.h>
|
||||||
|
#import <Foundation/Foundation.h>
|
||||||
|
#include <mach-o/dyld.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <stdint.h>
|
||||||
|
|
||||||
|
#define TAG @"[OQAdsLogger]"
|
||||||
|
|
||||||
|
%hook IMAAdsRequest
|
||||||
|
- (instancetype)initWithAdsResponse:(NSString *)adsResponse
|
||||||
|
adDisplayContainer:(id)adDisplayContainer
|
||||||
|
avPlayerVideoDisplay:(id)avPlayerVideoDisplay
|
||||||
|
pictureInPictureProxy:(id)pipProxy
|
||||||
|
userContext:(id)userContext {
|
||||||
|
|
||||||
|
NSLog(@"%@-[IMAAdsRequest init] [PiP path]", TAG);
|
||||||
|
NSLog(@"%@ adDisplayContainer = %@", TAG, adDisplayContainer);
|
||||||
|
NSLog(@"%@ avPlayerVideoDisplay = %@", TAG, avPlayerVideoDisplay);
|
||||||
|
NSLog(@"%@ pictureInPictureProxy = %@", TAG, pipProxy);
|
||||||
|
NSLog(@"%@ VMAP payload (%lu bytes):\n%@",
|
||||||
|
TAG, (unsigned long)adsResponse.length, adsResponse);
|
||||||
|
|
||||||
|
id result = %orig;
|
||||||
|
NSLog(@"%@ -> IMAAdsRequest = %p", TAG, result);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
- (instancetype)initWithAdsResponse:(NSString *)adsResponse
|
||||||
|
adDisplayContainer:(id)adDisplayContainer
|
||||||
|
contentPlayhead:(id)contentPlayhead
|
||||||
|
userContext:(id)userContext {
|
||||||
|
// Replace the VMAP with an empty document
|
||||||
|
NSString *emptyVMAP = @"<?xml version='1.0' encoding='utf-8'?>"
|
||||||
|
@"<vmap:VMAP xmlns:vmap=\"http://www.iab.net/vmap-1.0\" version=\"1.0\"/>";
|
||||||
|
NSLog(@"%@ Replaced VMAP (%lu bytes) with empty document", TAG, adsResponse.length);
|
||||||
|
return %orig(emptyVMAP, adDisplayContainer, contentPlayhead, userContext);
|
||||||
|
}
|
||||||
|
%end
|
||||||
|
|
||||||
|
// IMAAdsLoader — confirms dispatch to IMA SDK
|
||||||
|
//
|
||||||
|
// Called from OQPlayerAdsLoader.requestAds(_:loader:) (sub_10008FC08)
|
||||||
|
// after Swift dynamic-cast guards pass. This is the point of no return —
|
||||||
|
// the IMA SDK takes ownership of the request and starts network I/O.
|
||||||
|
|
||||||
|
%hook IMAAdsLoader
|
||||||
|
- (void)requestAdsWithRequest:(id)request {
|
||||||
|
NSLog(@"%@ -[IMAAdsLoader requestAdsWithRequest:]", TAG);
|
||||||
|
NSLog(@"%@ loader = %@", TAG, self);
|
||||||
|
NSLog(@"%@ request = %@", TAG, [request debugDescription]);
|
||||||
|
%orig;
|
||||||
|
NSLog(@"%@ requestAdsWithRequest dispatched ✓", TAG);
|
||||||
|
}
|
||||||
|
%end
|
||||||
|
|
||||||
|
// IMAAdsManager — SDK parsed the VMAP, breaks are scheduled
|
||||||
|
//
|
||||||
|
// Called from the IMAAdsLoaderDelegate callback in OQImaManager
|
||||||
|
// At this point the IMA SDK has parsed the VMAP and knows all ad break
|
||||||
|
// positions. Passing nil for renderingSettings means OQEE uses defaults.
|
||||||
|
|
||||||
|
%hook IMAAdsManager
|
||||||
|
- (void)initializeWithAdsRenderingSettings:(id)renderingSettings {
|
||||||
|
NSLog(@"%@ -[IMAAdsManager initializeWithAdsRenderingSettings:]", TAG);
|
||||||
|
NSLog(@"%@ adsManager = %@", TAG, self);
|
||||||
|
NSLog(@"%@ renderingSettings = %@", TAG, renderingSettings ?: @"(nil — default)");
|
||||||
|
%orig;
|
||||||
|
NSLog(@"%@ VMAP parsed, ad breaks scheduled ✓", TAG);
|
||||||
|
}
|
||||||
|
%end
|
||||||
|
|
||||||
|
|
||||||
|
%hook VSSubscriptionRegistrationCenter
|
||||||
|
- (void)setCurrentSubscription:(id)subscription
|
||||||
|
{
|
||||||
|
NSLog(@"Blocked VSSubscriptionRegistrationCenter");
|
||||||
|
NSLog(@"Subscription: %@", subscription);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
%end
|
||||||
|
|
||||||
|
%ctor {
|
||||||
|
// activate dev mode
|
||||||
|
NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
|
||||||
|
[defaults setBool:YES forKey:@"tv.oqee.devModeEnabled"];
|
||||||
|
}
|
||||||
9
OqeePlus/OqeePlus-tvOS/control
Normal file
9
OqeePlus/OqeePlus-tvOS/control
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
Package: xyz.nohamr.oqeeplus
|
||||||
|
Name: Oqee+ (tvOS)
|
||||||
|
Version: 1.0
|
||||||
|
Architecture: appletvos-arm64
|
||||||
|
Description: Oqee+ Ads blocker hook for tvOS
|
||||||
|
Maintainer: NohamR
|
||||||
|
Author: NohamR
|
||||||
|
Section: Tweaks
|
||||||
|
Depends: mobilesubstrate (>= 0.9.5000)
|
||||||
22
OqeePlus/OqeePlus-tvOS/index.md
Normal file
22
OqeePlus/OqeePlus-tvOS/index.md
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
# OqeePlus tvOS
|
||||||
|
|
||||||
|
Block ads initialization on Oqee.
|
||||||
|
|
||||||
|
- **App**: [OqeePlus : Streaming, TV en Direct](https://apps.apple.com/fr/app/free-tv/id1542614107)
|
||||||
|
- **Tested version**: 2.40
|
||||||
|
- **Target**: tvOS
|
||||||
|
|
||||||
|
## Build
|
||||||
|
|
||||||
|
```sh
|
||||||
|
make package FINALPACKAGE=1
|
||||||
|
```
|
||||||
|
|
||||||
|
## Inject
|
||||||
|
|
||||||
|
```sh
|
||||||
|
cyan -i oqeeplus.ipa \
|
||||||
|
-o oqeeplus_patched.ipa \
|
||||||
|
-f xyz.nohamr.oqeeplus_1.0_appletvos-arm64.deb \
|
||||||
|
-u
|
||||||
|
```
|
||||||
10
README.md
10
README.md
@@ -5,10 +5,18 @@ iOS tweaks built with [Theos](https://theos.dev), injected into IPAs via [cyan](
|
|||||||
## Tweaks
|
## Tweaks
|
||||||
|
|
||||||
| Tweak | App | Target |
|
| Tweak | App | Target |
|
||||||
| ------------------------------------------------------- | ---------------- | ------- |
|
| ------------------------------------------------------- | ---------------- | ---------- |
|
||||||
| [ServerCatPremium](ServerCatPremium/index.md) | ServerCat 1.30.0 | iOS 17+ |
|
| [ServerCatPremium](ServerCatPremium/index.md) | ServerCat 1.30.0 | iOS 17+ |
|
||||||
| [ServerCatPremium (legacy)](ServerCatPremium_/index.md) | ServerCat 1.6.4 | iOS 15 |
|
| [ServerCatPremium (legacy)](ServerCatPremium_/index.md) | ServerCat 1.6.4 | iOS 15 |
|
||||||
| [TextasticPro](TextasticPro/index.md) | Textastic 10.9.2 | iOS 18+ |
|
| [TextasticPro](TextasticPro/index.md) | Textastic 10.9.2 | iOS 18+ |
|
||||||
|
| [BusinessJB](BusinessJB/index.md) | Business 2.3.000 | iOS 15 |
|
||||||
|
| [CreditAgricoleJB](CreditAgricoleJB/index.md) | Ma Banque 47.0.0 | iOS 15.8.6 |
|
||||||
|
| [Infuse](Infuse/Infuse-tvOS/index.md) | Infuse 8.2.4 | tvOS 18.3 |
|
||||||
|
| [Infuse (iOS)](Infuse/Infuse-iOS/index.md) | Infuse 8.4.2 | iOS 18+ |
|
||||||
|
| [TF1+ (tvOS)](TF1Plus/TF1Plus-tvOS/index.md) | TF1+ 11.36.0 | tvOS |
|
||||||
|
| [TF1+ (iOS)](TF1Plus/TF1Plus-iOS/index.md) | TF1+ 11.36.0 | iOS 14+ |
|
||||||
|
| [OqeePlus (tvOS)](OqeePlus/OqeePlus-tvOS/index.md) | Oqee 2.40 | tvOS 18.3 |
|
||||||
|
| [OqeePlus (iOS)](OqeePlus/OqeePlus-iOS/index.md) | Oqee 2.40 | iOS 18+ |
|
||||||
|
|
||||||
## Build
|
## Build
|
||||||
|
|
||||||
|
|||||||
5
RMHook/index.md
Normal file
5
RMHook/index.md
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
# RMHook-iOS
|
||||||
|
|
||||||
|
A dynamic library injection tweak for the reMarkable iOS app, enabling connection to self-hosted [rmfakecloud](https://github.com/ddvk/rmfakecloud) servers.
|
||||||
|
|
||||||
|
See the [GitHub Repository](https://github.com/NohamR/RMHook-iOS) for full details, building instructions, and credits.
|
||||||
3
TF1Plus/TF1Plus-iOS/.gitignore
vendored
Normal file
3
TF1Plus/TF1Plus-iOS/.gitignore
vendored
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
.theos/
|
||||||
|
packages/
|
||||||
|
.DS_Store
|
||||||
12
TF1Plus/TF1Plus-iOS/Makefile
Normal file
12
TF1Plus/TF1Plus-iOS/Makefile
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
TARGET = iphone:latest:14.0
|
||||||
|
ARCHS = arm64 arm64e
|
||||||
|
INSTALL_TARGET_PROCESSES = mytf1
|
||||||
|
|
||||||
|
include $(THEOS)/makefiles/common.mk
|
||||||
|
|
||||||
|
TWEAK_NAME = TF1Plus
|
||||||
|
|
||||||
|
TF1Plus_FILES = Tweak.x
|
||||||
|
TF1Plus_CFLAGS = -fobjc-arc
|
||||||
|
|
||||||
|
include $(THEOS_MAKE_PATH)/tweak.mk
|
||||||
10
TF1Plus/TF1Plus-iOS/TF1Plus.plist
Normal file
10
TF1Plus/TF1Plus-iOS/TF1Plus.plist
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
{
|
||||||
|
Filter = {
|
||||||
|
Bundles = (
|
||||||
|
"com.tf1.applitf1",
|
||||||
|
);
|
||||||
|
Executables = (
|
||||||
|
App,
|
||||||
|
);
|
||||||
|
};
|
||||||
|
}
|
||||||
16
TF1Plus/TF1Plus-iOS/Tweak.x
Normal file
16
TF1Plus/TF1Plus-iOS/Tweak.x
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
#import <Foundation/Foundation.h>
|
||||||
|
|
||||||
|
%hook FWRequestConfiguration
|
||||||
|
- (id)initWithServerURL:(id)arg1 playerProfile:(id)arg2 {
|
||||||
|
return self;
|
||||||
|
}
|
||||||
|
%end
|
||||||
|
|
||||||
|
%hook VSSubscriptionRegistrationCenter
|
||||||
|
- (void)setCurrentSubscription:(id)subscription
|
||||||
|
{
|
||||||
|
NSLog(@"Blocked VSSubscriptionRegistrationCenter");
|
||||||
|
NSLog(@"Subscription: %@", subscription);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
%end
|
||||||
9
TF1Plus/TF1Plus-iOS/control
Normal file
9
TF1Plus/TF1Plus-iOS/control
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
Package: xyz.nohamr.tf1plus
|
||||||
|
Name: TF1+ (iOS)
|
||||||
|
Version: 1.0
|
||||||
|
Architecture: iphoneos-arm64
|
||||||
|
Description: TF1+ Ads blocker hook for iOS
|
||||||
|
Maintainer: NohamR
|
||||||
|
Author: NohamR
|
||||||
|
Section: Tweaks
|
||||||
|
Depends: mobilesubstrate (>= 0.9.5000)
|
||||||
22
TF1Plus/TF1Plus-iOS/index.md
Normal file
22
TF1Plus/TF1Plus-iOS/index.md
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
# TF1+ iOS
|
||||||
|
|
||||||
|
Block ads initialization on TF1+.
|
||||||
|
|
||||||
|
- **App**: [TF1+ : Streaming, TV en Direct](https://apps.apple.com/fr/app/tf1-streaming-tv-en-direct/id407248490)
|
||||||
|
- **Tested version**: 23.3.1
|
||||||
|
- **Target**: iOS
|
||||||
|
|
||||||
|
## Build
|
||||||
|
|
||||||
|
```sh
|
||||||
|
make package FINALPACKAGE=1
|
||||||
|
```
|
||||||
|
|
||||||
|
## Inject
|
||||||
|
|
||||||
|
```sh
|
||||||
|
cyan -i tf1plus.ipa \
|
||||||
|
-o tf1plus_patched.ipa \
|
||||||
|
-f xyz.nohamr.tf1plus_1.0_iphoneos-arm64.deb \
|
||||||
|
-u
|
||||||
|
```
|
||||||
3
TF1Plus/TF1Plus-tvOS/.gitignore
vendored
Normal file
3
TF1Plus/TF1Plus-tvOS/.gitignore
vendored
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
.theos/
|
||||||
|
packages/
|
||||||
|
.DS_Store
|
||||||
13
TF1Plus/TF1Plus-tvOS/Makefile
Normal file
13
TF1Plus/TF1Plus-tvOS/Makefile
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
TARGET = appletv:latest:18.3
|
||||||
|
ARCHS = arm64
|
||||||
|
INSTALL_TARGET_PROCESSES = mytf1
|
||||||
|
THEOS_PACKAGE_SCHEME = rootless
|
||||||
|
|
||||||
|
include $(THEOS)/makefiles/common.mk
|
||||||
|
|
||||||
|
TWEAK_NAME = TF1Plus
|
||||||
|
|
||||||
|
TF1Plus_FILES = Tweak.x
|
||||||
|
TF1Plus_CFLAGS = -fobjc-arc
|
||||||
|
|
||||||
|
include $(THEOS_MAKE_PATH)/tweak.mk
|
||||||
10
TF1Plus/TF1Plus-tvOS/TF1Plus.plist
Normal file
10
TF1Plus/TF1Plus-tvOS/TF1Plus.plist
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
{
|
||||||
|
Filter = {
|
||||||
|
Bundles = (
|
||||||
|
"com.tf1.applitf1",
|
||||||
|
);
|
||||||
|
Executables = (
|
||||||
|
mytf1,
|
||||||
|
);
|
||||||
|
};
|
||||||
|
}
|
||||||
7
TF1Plus/TF1Plus-tvOS/Tweak.x
Normal file
7
TF1Plus/TF1Plus-tvOS/Tweak.x
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
#import <Foundation/Foundation.h>
|
||||||
|
|
||||||
|
%hook FWRequestConfiguration
|
||||||
|
- (id)initWithServerURL:(id)arg1 playerProfile:(id)arg2 {
|
||||||
|
return self;
|
||||||
|
}
|
||||||
|
%end
|
||||||
9
TF1Plus/TF1Plus-tvOS/control
Normal file
9
TF1Plus/TF1Plus-tvOS/control
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
Package: xyz.nohamr.tf1plus
|
||||||
|
Name: TF1+ (Rootless)
|
||||||
|
Version: 1.0
|
||||||
|
Architecture: appletvos-arm64
|
||||||
|
Description: TF1+ Ads blocker hook
|
||||||
|
Maintainer: NohamR
|
||||||
|
Author: NohamR
|
||||||
|
Section: Tweaks
|
||||||
|
Depends: firmware
|
||||||
22
TF1Plus/TF1Plus-tvOS/index.md
Normal file
22
TF1Plus/TF1Plus-tvOS/index.md
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
# TF1+ tvOS
|
||||||
|
|
||||||
|
Block ads initialization on TF1+.
|
||||||
|
|
||||||
|
- **App**: [TF1+ : Streaming, TV en Direct](https://apps.apple.com/fr/app/tf1-streaming-tv-en-direct/id407248490)
|
||||||
|
- **Tested version**: 11.36.0
|
||||||
|
- **Target**: tvOS
|
||||||
|
|
||||||
|
## Build
|
||||||
|
|
||||||
|
```sh
|
||||||
|
make package FINALPACKAGE=1
|
||||||
|
```
|
||||||
|
|
||||||
|
## Inject
|
||||||
|
|
||||||
|
```sh
|
||||||
|
cyan -i tf1plus.ipa \
|
||||||
|
-o tf1plus_patched.ipa \
|
||||||
|
-f com.yourname.tf1plus_1.0_tvos-arm64.deb \
|
||||||
|
-u
|
||||||
|
```
|
||||||
BIN
docs/screens/BusinessJB/jailbreak.png
Normal file
BIN
docs/screens/BusinessJB/jailbreak.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 282 KiB |
BIN
docs/screens/CreditAgricoleJB/undetected.png
Normal file
BIN
docs/screens/CreditAgricoleJB/undetected.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 833 KiB |
BIN
docs/screens/Infuse/mobile.png
Normal file
BIN
docs/screens/Infuse/mobile.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 232 KiB |
BIN
docs/screens/Infuse/settings.png
Normal file
BIN
docs/screens/Infuse/settings.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 1.1 MiB |
71
scripts/patch_and_serve.sh
Executable file
71
scripts/patch_and_serve.sh
Executable file
@@ -0,0 +1,71 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
set -e
|
||||||
|
|
||||||
|
if [ "$#" -ne 2 ]; then
|
||||||
|
echo "Usage: $0 <file.ipa> <file.deb>"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
IPA=""
|
||||||
|
DEB=""
|
||||||
|
|
||||||
|
for arg in "$@"; do
|
||||||
|
case "$arg" in
|
||||||
|
*.ipa)
|
||||||
|
IPA="$arg"
|
||||||
|
;;
|
||||||
|
*.deb)
|
||||||
|
DEB="$arg"
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
echo "Unknown file type: $arg"
|
||||||
|
exit 1
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
done
|
||||||
|
|
||||||
|
if [ -z "$IPA" ] || [ -z "$DEB" ]; then
|
||||||
|
echo "You must provide one .ipa and one .deb file."
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
# ---- Prepare output folder ----
|
||||||
|
OUT_DIR="/tmp/ipa_patched"
|
||||||
|
mkdir -p "$OUT_DIR"
|
||||||
|
|
||||||
|
IPA_NAME=$(basename "$IPA")
|
||||||
|
OUTPUT_IPA="$OUT_DIR/$IPA_NAME"
|
||||||
|
|
||||||
|
echo "[+] Patching IPA with cyan..."
|
||||||
|
cyan -i "$IPA" -o "$OUTPUT_IPA" -f "$DEB" -u --overwrite
|
||||||
|
echo "[+] Patch complete."
|
||||||
|
|
||||||
|
LOCAL_IP=$(ipconfig getifaddr en0 2>/dev/null)
|
||||||
|
|
||||||
|
if [ -z "$LOCAL_IP" ]; then
|
||||||
|
LOCAL_IP=$(hostname -I 2>/dev/null | awk '{print $1}')
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ -z "$LOCAL_IP" ]; then
|
||||||
|
echo "Could not detect local IP automatically."
|
||||||
|
LOCAL_IP="YOUR_IP"
|
||||||
|
fi
|
||||||
|
|
||||||
|
DOWNLOAD_LINK="http://$LOCAL_IP:8000/$IPA_NAME"
|
||||||
|
|
||||||
|
cd "$OUT_DIR"
|
||||||
|
|
||||||
|
echo ""
|
||||||
|
echo "=========================================="
|
||||||
|
echo "Download link:"
|
||||||
|
echo "$DOWNLOAD_LINK"
|
||||||
|
echo "=========================================="
|
||||||
|
echo ""
|
||||||
|
echo -n "$DOWNLOAD_LINK" | pbcopy
|
||||||
|
echo "[+] Download link copied to clipboard."
|
||||||
|
echo "[+] Starting HTTP server..."
|
||||||
|
echo "Press Ctrl+C to stop."
|
||||||
|
echo ""
|
||||||
|
|
||||||
|
python3 -m http.server 8000
|
||||||
128
scripts/patch_and_server.py
Normal file
128
scripts/patch_and_server.py
Normal file
@@ -0,0 +1,128 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
|
import sys
|
||||||
|
import os
|
||||||
|
import subprocess
|
||||||
|
import shutil
|
||||||
|
import socket
|
||||||
|
|
||||||
|
|
||||||
|
def get_local_ip():
|
||||||
|
try:
|
||||||
|
# Try macOS command
|
||||||
|
output = subprocess.check_output(
|
||||||
|
["ipconfig", "getifaddr", "en0"], stderr=subprocess.DEVNULL
|
||||||
|
)
|
||||||
|
return output.decode("utf-8").strip()
|
||||||
|
except subprocess.CalledProcessError:
|
||||||
|
try:
|
||||||
|
# Try Linux command
|
||||||
|
output = subprocess.check_output(
|
||||||
|
["hostname", "-I"], stderr=subprocess.DEVNULL
|
||||||
|
)
|
||||||
|
return output.decode("utf-8").split()[0].strip()
|
||||||
|
except (subprocess.CalledProcessError, IndexError):
|
||||||
|
pass
|
||||||
|
|
||||||
|
# Fallback to socket method
|
||||||
|
try:
|
||||||
|
s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
|
||||||
|
s.connect(("8.8.8.8", 80))
|
||||||
|
ip = s.getsockname()[0]
|
||||||
|
s.close()
|
||||||
|
return ip
|
||||||
|
except Exception:
|
||||||
|
return None
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
if len(sys.argv) != 3:
|
||||||
|
print(f"Usage: {os.path.basename(sys.argv[0])} <file.ipa> <file.deb>")
|
||||||
|
sys.exit(1)
|
||||||
|
|
||||||
|
ipa_file = None
|
||||||
|
deb_file = None
|
||||||
|
|
||||||
|
for arg in sys.argv[1:]:
|
||||||
|
if arg.endswith(".ipa"):
|
||||||
|
ipa_file = arg
|
||||||
|
elif arg.endswith(".deb"):
|
||||||
|
deb_file = arg
|
||||||
|
else:
|
||||||
|
print(f"Unknown file type: {arg}")
|
||||||
|
sys.exit(1)
|
||||||
|
|
||||||
|
if not ipa_file or not deb_file:
|
||||||
|
print("You must provide one .ipa and one .deb file.")
|
||||||
|
sys.exit(1)
|
||||||
|
|
||||||
|
out_dir = "/tmp/ipa_patched"
|
||||||
|
os.makedirs(out_dir, exist_ok=True)
|
||||||
|
|
||||||
|
ipa_name = os.path.basename(ipa_file)
|
||||||
|
output_ipa = os.path.join(out_dir, ipa_name)
|
||||||
|
|
||||||
|
print("[+] Patching IPA with cyan...")
|
||||||
|
try:
|
||||||
|
subprocess.run(
|
||||||
|
[
|
||||||
|
"cyan",
|
||||||
|
"-i",
|
||||||
|
ipa_file,
|
||||||
|
"-o",
|
||||||
|
output_ipa,
|
||||||
|
"-f",
|
||||||
|
deb_file,
|
||||||
|
"-u",
|
||||||
|
"--overwrite",
|
||||||
|
],
|
||||||
|
check=True,
|
||||||
|
)
|
||||||
|
except subprocess.CalledProcessError:
|
||||||
|
print("[-] Patch failed.")
|
||||||
|
sys.exit(1)
|
||||||
|
except FileNotFoundError:
|
||||||
|
print(
|
||||||
|
"[-] 'cyan' command not found. Please ensure it is installed and in your PATH."
|
||||||
|
)
|
||||||
|
sys.exit(1)
|
||||||
|
|
||||||
|
print("[+] Patch complete.")
|
||||||
|
|
||||||
|
local_ip = get_local_ip()
|
||||||
|
if not local_ip:
|
||||||
|
print("Could not detect local IP automatically.")
|
||||||
|
local_ip = "YOUR_IP"
|
||||||
|
|
||||||
|
download_link = f"http://{local_ip}:8000/{ipa_name}"
|
||||||
|
|
||||||
|
print()
|
||||||
|
print("==========================================")
|
||||||
|
print("Download link:")
|
||||||
|
print(download_link)
|
||||||
|
print("==========================================")
|
||||||
|
print()
|
||||||
|
|
||||||
|
# Try to copy to clipboard using pbcopy on macOS
|
||||||
|
try:
|
||||||
|
if shutil.which("pbcopy"):
|
||||||
|
subprocess.run(["pbcopy"], input=download_link.encode("utf-8"), check=True)
|
||||||
|
print("[+] Download link copied to clipboard.")
|
||||||
|
except Exception:
|
||||||
|
pass
|
||||||
|
|
||||||
|
print("[+] Starting HTTP server...")
|
||||||
|
print("Press Ctrl+C to stop.")
|
||||||
|
print()
|
||||||
|
|
||||||
|
# Start python HTTP server
|
||||||
|
try:
|
||||||
|
subprocess.run(
|
||||||
|
[sys.executable, "-m", "http.server", "8000", "--directory", out_dir]
|
||||||
|
)
|
||||||
|
except KeyboardInterrupt:
|
||||||
|
print("\nStopping HTTP server...")
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
main()
|
||||||
159
scripts/watch_and_server.py
Normal file
159
scripts/watch_and_server.py
Normal file
@@ -0,0 +1,159 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
|
import sys
|
||||||
|
import os
|
||||||
|
import subprocess
|
||||||
|
import shutil
|
||||||
|
import socket
|
||||||
|
import time
|
||||||
|
import glob
|
||||||
|
import threading
|
||||||
|
|
||||||
|
|
||||||
|
def get_local_ip():
|
||||||
|
try:
|
||||||
|
output = subprocess.check_output(
|
||||||
|
["ipconfig", "getifaddr", "en0"], stderr=subprocess.DEVNULL
|
||||||
|
)
|
||||||
|
return output.decode("utf-8").strip()
|
||||||
|
except subprocess.CalledProcessError:
|
||||||
|
try:
|
||||||
|
output = subprocess.check_output(
|
||||||
|
["hostname", "-I"], stderr=subprocess.DEVNULL
|
||||||
|
)
|
||||||
|
return output.decode("utf-8").split()[0].strip()
|
||||||
|
except (subprocess.CalledProcessError, IndexError):
|
||||||
|
pass
|
||||||
|
|
||||||
|
try:
|
||||||
|
s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
|
||||||
|
s.connect(("8.8.8.8", 80))
|
||||||
|
ip = s.getsockname()[0]
|
||||||
|
s.close()
|
||||||
|
return ip
|
||||||
|
except Exception:
|
||||||
|
return None
|
||||||
|
|
||||||
|
|
||||||
|
def patch_ipa(ipa_file, deb_file, output_ipa):
|
||||||
|
print(f"\n[+] Patching IPA with cyan using {os.path.basename(deb_file)}...")
|
||||||
|
try:
|
||||||
|
subprocess.run(
|
||||||
|
[
|
||||||
|
"cyan",
|
||||||
|
"-i",
|
||||||
|
ipa_file,
|
||||||
|
"-o",
|
||||||
|
output_ipa,
|
||||||
|
"-f",
|
||||||
|
deb_file,
|
||||||
|
"-u",
|
||||||
|
"--overwrite",
|
||||||
|
],
|
||||||
|
check=True,
|
||||||
|
)
|
||||||
|
print("[+] Patch complete.")
|
||||||
|
return True
|
||||||
|
except subprocess.CalledProcessError:
|
||||||
|
print("[-] Patch failed.")
|
||||||
|
return False
|
||||||
|
except FileNotFoundError:
|
||||||
|
print(
|
||||||
|
"[-] 'cyan' command not found. Please ensure it is installed and in your PATH."
|
||||||
|
)
|
||||||
|
return False
|
||||||
|
|
||||||
|
|
||||||
|
def get_newest_deb(directory):
|
||||||
|
deb_files = glob.glob(os.path.join(directory, "*.deb"))
|
||||||
|
if not deb_files:
|
||||||
|
return None
|
||||||
|
return max(deb_files, key=os.path.getmtime)
|
||||||
|
|
||||||
|
|
||||||
|
def start_server(out_dir):
|
||||||
|
server_process = subprocess.Popen(
|
||||||
|
[sys.executable, "-m", "http.server", "8000", "--directory", out_dir]
|
||||||
|
)
|
||||||
|
return server_process
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
if len(sys.argv) != 3:
|
||||||
|
print(f"Usage: {os.path.basename(sys.argv[0])} <file.ipa> <file.deb>")
|
||||||
|
sys.exit(1)
|
||||||
|
|
||||||
|
ipa_file = None
|
||||||
|
initial_deb_file = None
|
||||||
|
|
||||||
|
for arg in sys.argv[1:]:
|
||||||
|
if arg.endswith(".ipa"):
|
||||||
|
ipa_file = arg
|
||||||
|
elif arg.endswith(".deb"):
|
||||||
|
initial_deb_file = arg
|
||||||
|
else:
|
||||||
|
print(f"Unknown file type: {arg}")
|
||||||
|
sys.exit(1)
|
||||||
|
|
||||||
|
if not ipa_file or not initial_deb_file:
|
||||||
|
print("You must provide one .ipa and one .deb file.")
|
||||||
|
sys.exit(1)
|
||||||
|
|
||||||
|
out_dir = "/tmp/ipa_patched"
|
||||||
|
os.makedirs(out_dir, exist_ok=True)
|
||||||
|
|
||||||
|
ipa_name = os.path.basename(ipa_file)
|
||||||
|
output_ipa = os.path.join(out_dir, ipa_name)
|
||||||
|
|
||||||
|
deb_dir = os.path.dirname(os.path.abspath(initial_deb_file))
|
||||||
|
current_deb_file = get_newest_deb(deb_dir) or initial_deb_file
|
||||||
|
last_mtime = (
|
||||||
|
os.path.getmtime(current_deb_file) if os.path.exists(current_deb_file) else 0
|
||||||
|
)
|
||||||
|
|
||||||
|
if not patch_ipa(ipa_file, current_deb_file, output_ipa):
|
||||||
|
sys.exit(1)
|
||||||
|
|
||||||
|
local_ip = get_local_ip() or "YOUR_IP"
|
||||||
|
download_link = f"http://{local_ip}:8000/{ipa_name}"
|
||||||
|
|
||||||
|
print()
|
||||||
|
print("==========================================")
|
||||||
|
print("Download link:")
|
||||||
|
print(download_link)
|
||||||
|
print("==========================================")
|
||||||
|
print()
|
||||||
|
|
||||||
|
try:
|
||||||
|
if shutil.which("pbcopy"):
|
||||||
|
subprocess.run(["pbcopy"], input=download_link.encode("utf-8"), check=True)
|
||||||
|
print("[+] Download link copied to clipboard.")
|
||||||
|
except Exception:
|
||||||
|
pass
|
||||||
|
|
||||||
|
print("[+] Starting HTTP server...")
|
||||||
|
server_process = start_server(out_dir)
|
||||||
|
|
||||||
|
print(f"[+] Watching for new .deb files in {deb_dir} every 2 seconds...")
|
||||||
|
print("Press Ctrl+C to stop.")
|
||||||
|
|
||||||
|
try:
|
||||||
|
while True:
|
||||||
|
time.sleep(2)
|
||||||
|
newest_deb = get_newest_deb(deb_dir)
|
||||||
|
if newest_deb:
|
||||||
|
current_mtime = os.path.getmtime(newest_deb)
|
||||||
|
if current_mtime > last_mtime:
|
||||||
|
print(
|
||||||
|
f"\n[*] Detected new/updated .deb file: {os.path.basename(newest_deb)}"
|
||||||
|
)
|
||||||
|
patch_ipa(ipa_file, newest_deb, output_ipa)
|
||||||
|
last_mtime = current_mtime
|
||||||
|
except KeyboardInterrupt:
|
||||||
|
print("\nStopping HTTP server and watcher...")
|
||||||
|
server_process.terminate()
|
||||||
|
server_process.wait()
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
main()
|
||||||
Reference in New Issue
Block a user