Android
Check out my talk and its resources about Android application hacking here
Android OS and specificities
Android Security Model
Two distinct layes to Android Security model.
1. Implemented in the OS, and isolates installed app from one another
Each app has a specific UID, dynamically assigned.
An app can only access its UID files and no other (except if shared by another app or OS)
Each App runs as a separate process under a sperate UID
Prior to Android 4.3 = the only thing that was isolating apps = if root compromised entire system was compromised
Starting from Android 4.3 = SELinux
SELinux denies all process interaction + create policies to allow only expected applications
2. Security of an App itself (made by the developers)
The dev can selectevely expose certain app functions to other apps
Configures App capabilities
All apps are in the /data/data folder (except if modified in manifest by dev)
The permissions declared in the manifest will be translated in permissions in the file system.
Build an app and fundamentals on Android Applications
Structure
An Android App comprises two main elements:
The program's core functionality, written in Java code or Kotlin (official language today)
The XML files that specify various configurations, including string values and the app's identity.
The manifest from a developer perspective
When an attribute starts with a "." = relative to package name so if package name is
com.example.myapp
an element attribute with.something
will becom.example.myapp.something
minSdkVersion
default value of 1 (app can run on any Android version, not ideal in a security point of view)targetSdkVersion
the version of Android the app has been tested on and is compatible with. (default will be the value of minSdkVersion)maxSdkVersion
usually not set because it will prevent the app from being installed on latest Android device.
Intents
An Intent is a messaging object that is used to communicate between components in an Android app, such as activities, services, and broadcast receivers.
Intents can be used to start new activities, pass data between activities or services, and broadcast events to other components in the system. They can also be used to start external activities or services provided by other apps, and to receive broadcasts from the system or other apps.
An Intent is essentially a way for an app to request an action from the Android system or other apps, or to respond to actions that are requested by other apps or the system. When an Intent is received, the component that handles it can extract any data that is passed with the Intent and use it to perform a specific action or provide a specific service to the user.
Implicit Intents
An Implicit Intent is a type of Intent in Android that does not explicitly specify the target component (i.e., the activity, service or broadcast receiver) to which the intent is intended to be delivered.
Instead, it declares an action to be performed, and optionally, data to be passed. The Android system then searches for all components that have registered to handle that specific action and data, and displays a list of appropriate apps that can handle the intent, allowing the user to choose which app to use.
For example, if you want to open a PDF file in your app, you can create an implicit intent that declares the "ACTION_VIEW" action and specifies the "application/pdf" data type. When the user selects a PDF file in a file manager app, the Android system will search for all components that can handle the "ACTION_VIEW" action and "application/pdf" data type. If your app has registered to handle this type of intent, it will be listed as an option for the user to select.
Implicit intents can be used to start activities, services, or broadcast receivers in other apps or the system, as well as to share data between apps. They are a powerful way to provide seamless integration between different apps and services in the Android ecosystem.
It is a string that will look like this ACTION_ACTION-TO-PERFORM
(for example VIEW
or SEND
). To specify an intent's action we use the setAction() method or the Intent Constructor.
To specify an Intent's action in Android, you need to call the setAction()
method on the Intent object and pass in a string that represents the action you want to perform. The action string specifies what type of action you want to perform, such as viewing a webpage or sending an email.
For example, to create an Intent that will open a webpage in the default web browser, you would create a new Intent object and call the setAction()
method with the ACTION_VIEW
constant from the Intent
class, like this:
This sets the action of the Intent to ACTION_VIEW
, which tells the Android system that you want to view something. You can then add additional information to the intent, such as the URL of the webpage you want to open, using methods like setData()
or putExtra()
.
Other common actions that you can specify in an Intent include ACTION_SEND
for sharing data, ACTION_EDIT
for editing data, and ACTION_DIAL
for dialing a phone number.
⚠️ Using implicit intents can potentially introduce some security risks in an Android app. When you use an implicit intent, you are not specifying the exact component that should receive the intent. Instead, the Android system searches for all components that can handle the intent's action and data type and presents the user with a list of possible options to choose from. This means that it is possible for a malicious app to register itself as a handler for a specific action and data type, and intercept the intent before it is delivered to the intended recipient. This could potentially allow the malicious app to access sensitive data or perform other unwanted actions.
Extra
An "extra" is a key-value pair that can be included in an Intent object to pass additional data between different components in an app. Extras are used to provide more detailed information about the action that the Intent is requesting, or to provide additional data that the receiving component needs in order to complete its task.
For example, if you are creating an Intent to launch a new activity and you want to pass some data to the new activity, you can use the putExtra() method to add an extra to the Intent. The extra can be any type of data that can be serialized, such as a string, integer, or boolean value.
Here's an example of how to add an extra to an Intent:
In this example, the putExtra()
method is used to add a string extra to the Intent with the key "EXTRA_KEY" and the value "extra value". The receiving component can then extract the extra data from the Intent by calling the appropriate method, such as getStringExtra()
.
Intent Filter
An Intent Filter is an element that you can define to declare the types of intents that your app can handle. It specifies the combination of action, data, and category values that an intent must have in order for the Android system to launch an activity, service, or broadcast receiver in your app. When an implicit intent is fired, the Android system searches for all the activities, services or broadcast receivers that have registered their intent filters for the specific action, data, and category in the intent. It then displays a list of the appropriate apps that can handle the intent, and allows the user to choose which app to use.
Here is an example
Suppose that the user clicks on a link to https://www.example.com/some-page
in a web browser on their Android device. The web browser will send an ACTION_VIEW intent to the Android system with the URL of the clicked link as data. The Android system will look for any installed apps that can handle the ACTION_VIEW intent and have an intent-filter that matches the data in the intent.
In this case, the MyActivity class has an intent-filter that matches the ACTION_VIEW intent with data that has an https scheme. Therefore, the Android system will launch the MyActivity class and pass the ACTION_VIEW intent to it. The MyActivity class can then extract the URL data from the intent and display the appropriate content to the user.
Priority
The priority attribute is an optional attribute that can be used in the <intent-filter>
element of an Android manifest file to specify the priority of an activity or service component relative to other components that can handle the same type of Intent. The priority is used by the Android system to determine which component should receive the Intent if there are multiple components that are capable of handling it.
The priority attribute is specified as an integer value between -1000 and 1000, with higher values indicating a higher priority. The default priority is set to 0 if the attribute is not specified.
For example, if you have two activities that can handle the same type of Intent, but you want one of the activities to be given priority over the other, you can specify a higher priority value for that activity using the android:priority
attribute. Here's an example of how to set the priority of an activity:
In this example, the android:priority
attribute is set to 100 for the MainActivity component, which indicates that it has a higher priority than other components that can handle the same type of Intent. This means that if there are multiple components that can handle the Intent, the Android system will give preference to the MainActivity component.
SYSTEM_HIGH_PRIORITY
SYSTEM_HIGH_PRIORITY
is a special priority constant that can be used with the android:priority
attribute in the <intent-filter>
element of an Android manifest file. This priority is reserved for system-level components that require the highest level of priority for handling certain types of Intents.
Components that have a priority set to SYSTEM_HIGH_PRIORITY
are given priority over all other components, including those with a priority of HIGH_PRIORITY
or DEFAULT_PRIORITY
. This means that if there are multiple components that can handle the same type of Intent, the component with the SYSTEM_HIGH_PRIORITY
priority will always be chosen.
This priority is typically reserved for system-level components that are critical for the functioning of the Android system, such as device administrators, accessibility services, and other system-level services. Using this priority for non-system-level components is discouraged, as it can negatively impact the performance and stability of the device.
If there is a conflict the user will have to choose the application through a popup window.
Here is an interesting vulnerability related to this. Using the SYSTEM_HIGH_PRIORITY it was possible "to impersonate the Google Play billing service and circumvent the signature verification".
Explicit Intents
An explicit intent is an Android Intent that specifies the exact component to be invoked by the Android system, such as an Activity, Service or BroadcastReceiver, to perform a specific action. With an explicit intent, you provide the full package name and class name of the component you want to launch.
For example, you can launch an activity with an explicit intent like this:
In this example, the Intent constructor takes two arguments: the current context (in this case, the activity that is launching the new activity), and the class of the activity to be launched. The main difference between explicit and implicit intents is that an explicit intent targets a specific component while an implicit intent is broadcast to all components that have registered to handle the specified action.
Broadcast Intent
A broadcast intent is a type of Android Intent that is used to broadcast messages or events to multiple components within the Android system, such as activities, services, and broadcast receivers. Broadcast intents are a mechanism for inter-component communication that allows one component to send a message or event to multiple other components at once, without knowing the specific components that will receive the message.
Broadcast intents can be sent by both system-level components and user-level applications, and can be used for a wide variety of purposes, such as notifying other components of events, triggering system-level actions, and passing data between components.
There are two types of broadcast intents in Android:
Normal broadcasts: These are asynchronous broadcasts that are delivered to all registered receivers in an unspecified order, regardless of their priority or timing. Normal broadcasts are usually used for events that do not require immediate action, such as notifications or updates.
Ordered broadcasts: These are synchronous broadcasts that are delivered to all registered receivers in a specific order, based on their priority and the order in which they were registered. Ordered broadcasts are usually used for events that require a specific sequence of actions, such as controlling the volume or brightness of the device.
Overall, broadcast intents are a powerful and flexible mechanism for inter-component communication in Android, and are widely used in both system-level and user-level applications.
See here the sendBroadcast method from the LocalBroadCastManager class. We can use it for components of the same app. This method is different from the sendBroadcast method from the Context class.
To secure broadcast intents we can use
Intent.setPackage
,uses-permission
Sticky broadcast
A sticky broadcast is a special type of broadcast in Android that stays "sticky" or persistent, even after the broadcast has been completed. When a sticky broadcast is sent, it is held in memory by the Android system, and any new receivers that register for that broadcast will immediately receive the last broadcast that was sent, even if it was sent before the receiver was registered.
Sticky broadcasts are useful in situations where an app wants to notify other components about a change in system state, such as when the battery level changes or the device is connected to a new network. By sending a sticky broadcast, an app can ensure that any newly registered receivers will immediately receive the last known state of the system, without having to wait for the next broadcast to be sent.
However, it is important to note that the use of sticky broadcasts should be avoided whenever possible, as they can be a security risk and can cause performance issues on the device. In fact, starting from Android 8.0 (API level 26), the use of sticky broadcasts is no longer allowed for most apps, except for a few system-level broadcasts. Instead, apps should use other mechanisms, such as foreground services or the JobScheduler API, to receive notifications about changes in system state. More info on this here
⚠️ When auditing an app it is worth checking for the keyword "sticky" during the static analysis.
Lab setup
You can use a rooted Android device or an AVD (Android Virtual Device)
My talk here for TDI covers the lab setup with an AVD
Get the app
Ask your customer to give it to you with SSL Pinning disabled. This is the best option...
Google play (we can use apk pull and take it from an AVD or device you will need an account on Google)
Emulator
Emulator is the command line tool that will allow you to manage Android virtual device. It gets installed when you install Android Studio.
If you are like me and prefer to use command line over GUI you can use it
If you do not know where the binary is you can run
find / -name emulator 2>/dev/null
./emulator -list-avds
will list the available VM./emulator -avd vm-name
will launch a specific VM
Device rooting
Rooting a device often requires to exploit a vulnerability on a device. This vulnerability needs to be in a process with system level access.
We could also use one exploit to gain access and another one for privilege escalation.
One exploit is done, the su binary has to be specified in the user's PATH environment variable like
system/xbin
. It then needs to be configured with the proper permissionsFinally we can use an app to interface with the su binary and process requests for root access.
⚠️ Attention: We have to be careful when rooting a device, because some apps used to root a device might be malicious, not properly coded or insecure. ⚠️ This also means that any app on the device could request root access. This obviously exposes the device and the data on it.
Custom ROMs
It is possible to replace the OS that comes with a device with custom firmware. It can be convenient to keep an older device longer, bypass software restrictions (OEM) or get latest Android code without having to wait for an update. This requires to root the device to modify the bootloader so that it can load the custom ROM firmware. Sometimes it is even allowed by some manufacturers (Google Nexus for example). This is safer and easier with these devices so they are a good option for pentesters.
Note Nexus devices do not come with carrier bloatware pre-installed.
ADB
Install apk
Once the emulator is launched we can use adb with
adb shell
Install without adb shell:
The application can then be found in the app list
We can also drag and drop the apk in the emulator
ADB commands
adb shell
- Connect to deviceadb install
- sideload apksadb push
- PC to deviceadb pull path/name.apk
- Device to PCadb logcat
- commande line took that dumps a log of system messagesadb shell pm list packages
will give you a list of all installed package namesadb shell pm list packages -f 3
will give you a list of all 3rd party installed package names. Both list packages commands will also give you the path, so it will be convenient to pull itadb uninstall <package-name>
uninstall a packageLaunch activities (they should be exported)
Find the arch of your android Device or emulator
This is useful to know which version of frida server you need for example.
adb shell getprop
To figure out what kind of device you have:
Check this thread on stack exchange for more details
or
adb shell getprop ro.product.cpu.abi
ADB troubleshoot
If like me you pulled your hair because of Android Dolphin emulator and could not find proxy settings, you can get back the former look of the emulator by changing this in the settings
Thanks to Inliner on stakoverflow for this solution. You can find the thread here
Burp
Getting burp cert and push it with adb
Open Burp Suite go to the proxy tab. Then go to the Options tab click on "Import / export CA certificate"
Select Certificate in DER format
Push the certificate to your device
adb push path/to/my/cert.der /data/local/tmp/cert-der.crt
You can also drag and drop it if you use and emulator
Configure Burp with Android Studio
Decompile code
launch with
java -jar
and open apk
jadx-gui
APKLab (add on visual studio Code)
MobSF
Install Mobsf
Launch mobsf
MobSF will listen to 0.0.0.0:8000 if you run the script without arguments.
Mobsf with Docker
If you want you can use docker to get and use MobSF
docker pull opensecurity/mobile-security-framework-mobsf
docker run -it -p 8000:8000 opensecurity/mobile-security-framework-mobsf
Got to
http://0.0.0.0:8000
to access the gui
Note: You won't be able to use dynamic analysis.
Official Documentation
Mobsf nice features
Very helpful for static analysis
Will look into the source code and find secrets
You can also search in the code for specific strings
Checklists
On every pentest, it is always worth having a checklist. This will allow you to make sure that you do not forget anything. For Android and Mobile Application pentest in general, I can only recommend the MAS checklist made by OWASP.
Check it out here
You can also use MindAPI by David Sopas
Static Analysis
Android manifest
It is located in "resources". This file is where developers will describe the essential information of the application. As stated in the Android documentation for developers: "Every application project must have an AndroidManifest.xml file (with this exact name) at the root of the project's source set. The manifest file describes your app's essential information to the Android build tools, the Android operating system and Google Play." See Reference
It will be used to mention the application package name, the application components, the permissions and the hardware and software that the application needs. Note that it is also useful to check which API we need to test the application by looking at the minSDKVersion
.
Since the purpose of permissions is to protect the privacy of the user, you can see here some permissions that should not be allowed, depending of course on the purpose of the application.
See here full list of permissions.
When analyzing the manifest you can also determine how the application retrieves files.
It is worth noting that during the development process developer might have multiple manifest files. They will need to merge them. See here how it is done
Allow backup
The allowBackup flag could allow an attacker to take the backup of the application data with adb. So it would be safer to set it to false, as mentioned in the owasp documentation if this attribute is not set, it is enabled by default. More details: OWASP-MTG
Debuggable
This flag indicates if the application can be debugged. It must be set to false otherwise an attacker could use it to access the application data. Be aware that sometimes your client will give you the developer version of the application, so this flag will be set to true. What I usually do in this case is I set it as informative to remind the customer to make sure they change this when they commit the application to production.
Here is what the flag looks like:
android:debuggable="true"
See documentation OWASP MTG about this.
Exportable activities
Activities are the screens of the applications. Depending on the application and the activity, some of them should not be exportable because it means that they could be accessible from outside the application. It means that you could execute it with abd without going through the main activity. Here is an example of code on the pivaa application with exportable activities:
Activities
Activities are always worth checking. Sometimes you might see activities that takes command line arguments.
For confidentiality reasons the examples here are anonymized
So it might be a good idea in the Activity files to look for the following strings: "extra", "extras" "intent"
Methodology to analyze an activity
In the android manifest, find the activities.
This is what it looks like in the Android UnCrackable L1 Crackme from OWASP MAS.
Using these information you can check them with jadx-gui or apktool the activities
Sometimes in some application you will have a reference to the r.class this class contains the public constants used by application. This thread on stackoverflow is really interesting to know more about this class.
Note that you do not need specifically to go through the activity to analyze the r.class. It can happen that there will be no reference to the r.class, but you can still check it anyway
In this other example we have a reference to r.class in the activity:
setContentView(2130968602);
What we can do is copy the number, and look for it in the r.class (we can CTRL+F once in the r.class)
It will highlight in the code as follows
Now in the res/layout folder you decompiled with apktool, you can look for an xml file named
activity_access.xml
.It is worth checking if it has any interesing info. It is also going to help you understand how the app works. They are used for the view (layout) of an app.
It is interesting to check if it has "onClick" actions for the button or any other references to functions.
If we go back to our crackme example we do not have a reference in our file to the r.class, but if we check it anyway we can find this
So we can check in
layout>activity_main.xml
in our decompiled files what it looks like.So we could analyze
menu>menu_main.xml
Here is an example of something that would be worth analysing of
layout>activity_main.xml
. We can see the xml code corresponding to the button and that this code is calling a function verify.
General tips for static analysis
Check Strings.xml
Try to enumerate database
See Firebase Enum
Enumerate public cloud resources
See Cloud Enum
Lookup for: Secret keys, passwords, comments, URLs, IP, private keys, any sensitive information that should not be in the code. With Jadx-gui it is possible to use the global search to search for specific strings such as: API, API_KEY, password, key, ClientId, ClientSecret, id, AWS, Secret, username, firebase.io, http, https, SQL (or other SQL file extensions) See here how to use glbal search with jadx-gui:
An alternative to jadx-gui which is even better in my opinion is if you used mobsf you will have folder
/uploads
and you can grep in it for interesting strings. Check out Alissa Knight's video below, she mentions it.
When you find a database you can then try to read it with sqlitebrowser to see what is in it. See OWASP about this.
Interesting resources for static analysis on Hacktricks
Modify an Application
For pentest, CTF and other purposes (like cert pinning bypass) we might need to modify the code of the application.
However the only way to do so when we only have the apk provided is to decompile the code with apktool (or any other) modify the smali code to fit our purpose, recompile it and sign it.
Decompile with apktool
apktool d app.apk
decompile the package app.apk (the d is for decode)
To prevent the classes.dex file from being disassembled you can use
-s
flag
apktool.yml
This file will tell us the version of apktool, the name of the apk file, and if it uses a Framwork. We can also get the minimum and target sdk version.
A framework is a set of pre-written software components, tools, and libraries that provide a structure for developers to build their app upon, allowing for faster and more efficient development.
The SharedLibrary section specifies the shared libraries that the app requires in order to function properly. This information can be useful during the recompiling process to ensure that the required shared libraries are included in the rebuilt APK file.
Original directory
This directory contains the original resources and manifest file of the app, before any modifications were made during the packaging process.
smali directory
The "smali" directory is a subdirectory of the decompiled APK directory generated by Apktool when decompiling an APK file. It contains the app's bytecode translated into the Smali language, which is a human-readable and editable form of the Dalvik bytecode used by Android.
The Smali language allows developers to inspect and modify the app's code at a low level, such as changing method calls or modifying the flow of the code. This can be useful for debugging or reverse engineering purposes, but it should be noted that modifying an app's code without proper authorization or legal permission may violate intellectual property laws or terms of service.
It contains the individual smali files for all of the decoded dex files organized in directories based on the Java package name.
Unknown directory
Apktool may generate an "unknown" directory when decompiling an APK file, which contains any resources or assets that Apktool was unable to identify or parse during the decompilation process.
This may occur if the APK uses custom or proprietary resources or obfuscation techniques, or if the APK was corrupted or modified in a way that Apktool cannot handle. The "unknown" directory may contain binary files or other resources that cannot be easily edited or analyzed without specialized tools or knowledge.
What is an .odex file
An ODEX file is a file format used by the Android operating system to store pre-optimized bytecode for an Android application. When an APK file is installed on an Android device, the Dalvik VM converts the app's bytecode into a format that can be executed by the device's processor. By creating an ODEX file during installation, the Dalvik VM can load the optimized bytecode directly, which can improve the app's startup time and performance.
An ODEX file has the same name as the APK file but with an additional .odex extension, and it is typically located in the /data/dalvik-cache directory on the device's internal storage.
If you have and odex file it means you won't have a classes.dex.
An odex will speed up the first boot of a rom.
Understand smali
Registers
Source: Wiki smali/baksmali
Types
Source: Wiki smali/baksmali
Hello world
Here is an example of an hello world in smali. You can find the original code here
If you use arguments they have to be placed in the last parameters.
Smali tips to understand
The
getString()
function might use a controlled user input so it is always worth checking.
Smali/Baksmali
Dowload smali and baksmali here
java -jar baksmali.jar file.dex
disassemble a .dex filejava -jar smali.jar folder-with-smali-code/
generate dex file from smali code
Dex2jar
Dex2jar is an open source project to work with dex and class files. We can use dex2jar to convert dex or apk in jar. This way we can use java decompiler tools and we will get a more java looking view of the code.
dex2jar file.dex -o file.jar
convert dex to jardex2jar file.apk -o file.jar
convert apk to jar
Recompile with apktool
apktool b app/ -o newapp.apk
recompile the folderapp/
and generate a new filenewapp.apk
Sign the application
In order to be installed we need to sign the application. We first need a key
keytool -genkey -v -keystore my-release-key.keystore -alias alias_name -keyalg RSA -keysize 2048 -validity 10000
Now we can sign it. We can use apksigner.
apksigner sign --ks path/to/my-release-key.keystore newapp.apk
We could also use jarsigner
jarsigner -verbose -sigalg SHA1withRSA -digestalg SHA1 -keystore my-release-key.keystore my_application.apk alias_name
Security notes about keys: If a key from an Android App is compromised, the developers will have to sign new versions with a different key (this is how Google Play Store identify the org who signs the app). This could also allow an attacked to publish malicious apps which would be trusted because they were signed with a previously trusted key.
Signing modes
Debug mode is for testing purposes (you can run apps directly connected via USB or emulators)
Release mode is for production aka public release.
APK Alignment
zipalign -v 4 unaligned.apk aligned.apk
Improves RAM use when running the app.
OEM Apps
OEM apps on Android are pre-installed apps that come with the device and are developed and installed by the original equipment manufacturer (OEM). They can also be shipped with the Android Open Source Project (ASOP) or by a cell phone provider if you purchase the phone from them. These apps are often included as part of the device's firmware and cannot be uninstalled by the user without rooting the device.
OEM apps may include core system apps such as the dialer, messaging app, or camera app, as well as branded or customized apps that are specific to the device or manufacturer. In some cases, OEM apps may include bloatware (OEM apps, trial software, or other third-party apps that are included on the device by default) or unwanted apps that take up storage space and cannot be easily removed.
They can usually be found in /system/app
directory of the device.
It is common for many of them to run with systme (root level) permissions
Proguard
ProGuard is an open-source tool for code obfuscation, optimization, and shrinking for Java and Android applications. It is often used by developers to reduce the size of their app, remove unused code and resources, and make it more difficult for attackers to reverse engineer and analyze the app's code.
ProGuard can be integrated into the build process of an Android app and can be configured to perform various optimizations, such as removing unused code, renaming classes, methods, and fields to make the code harder to understand, and applying other code obfuscation techniques to protect the app from reverse engineering. ProGuard is typically used in conjunction with other tools, such as Android Studio or Gradle, to build and package the final APK file.
Dynamic analysis
General tips for dynamic analysis
Are screens visible in screenshots or instances with sensitive data? Understand: can I take a picture of a screen with sensitive data and if I switch to another application after being on a screen with sensitive data can I go back via cached instances?
All the usual OWASP Top 10 web vulns
Dynamic analysis on Hacktricks
Check if third party keyboard extensions are allowed and active keyboard cache on sensitive fields sensitive. Some malware masquerades as Android keyboard extensions.
Tapjacking PoC equivalent to Clickjacking can be done with Qark
How to bypass certificate pinning
Here is an article by Tim Perry on httptoolkit on how to bypass cert pinning with frida.
If none of the universal script work, you might want to consider using apktool to decompile, modify the smali code and recompile it (see how to do so above).
Disable SSL pinning with Frida
Intercept HTTPS traffic
Intercepting HTTPS Traffic from Apps on Android 7+ using Magisk & Burp this tutorial is great.
Here is how to install Magisk
If you want to modify just the network config using objection (and not injecting frida gadget). You can comment some code in objection. Comment the whole code of these 2 functions in objection in the file
/usr/local/lib/python3.10/dist-packages/objection/utils/patchers/android.py
To do so you can add this
'''
on line 765 and line 840 (this will comment the whole block of the 2 functions. Be careful with indentation to avoid errors)You will also need to comment 203 and 204 in
/usr/local/lib/python3.10/dist-packages/objection/commands/mobile_packages.py
because the previous 2 functions are called in this file.Finally run
objection patchapk -N -s application.apk
and patch the apk
Thanks to my colleague Ash for mentioning this to me.
If you want to patch the apk, and inject frida gadget this command should work
objection patchapk -s application.apk
. If you want a practical example of this check out the writeup for the challenged Anchored here.
Disable SSL pinning with objection
This article from Corsecure explains it really. You can also use the related video here To get further with Objection, you can check out hacktricks here.
Notes:
With an emulator you need to use the architecture flag. Example:
objection patchapk --source app.apk --architecture x86
Debug frida or objection
If you get the error
Unable to connect to the frida server: need Gadget to attach on jailed Android; its default location is: /home/user/.cache/frida/gadget-android-arm64.so
Get the gadget related to your to device and put it in the default location
If you get
Unable to connect to the frida server: command failed: 99
Instead of launching your app with
frida -U -l ./frida-script.js -f com.vendor.myapp
usefrida -U -l ./frida-script.js Appname
If frida freezes your vm. It seems to be this issue
What worked for me was to take the latest frida-server version (16.3.3)
Wireshark
Turn your laptop as hotspot capture traffic
Pull capture from a pfSense machine or something similar
adb shell tcpdump -s -s 0
HTTP traffic:
adb shell tcpdump -C -s -s 0 port 80
Medusa
Launch it with
medusa
Select your device (mine is 3)
After the selection you get the properties of the device and thr 3rd party installed app.
For more details check the wiki here
Objection
Here is the repo
Install:
pip3 install objection
Memory dump
Find the app you want to dump the memory from with
frida-ps -Uai
Hook on it with explore
objection --gadget com.mobilehackinglab.challenge explore
Dump the memory
memory dump all memory.txt
(memory.txt is the dump file you will get)
Fridump
If you encounter issues with objection for the memory dump, you can also use fridump.
Resources
My writeups for Android related boxes and challenges
More coming soon
Resources Android App
Courses on Android pentest
Lab Setup
References and reads
Practice
Vulnerable Apps to practice on
Hackthebox
Resources for SSL pinning bypass, Frida, traffic interception and rooting
Resources for Smali, Dalvik and RE an android App
Tools
List of tools
Android Emulator
Genymotion (free 30 days trial and free for personal use)
Android Studio (free)
Software
Automated tools
Certificate pinning and bypass - Tools
rootAVD by newb1t to root an AVD (I DID NOT TRY IT)
Magisk Trust User Certs (I did not try either it but it is suppose to be added in addition of use of rootAVD to make a system trusted certificate)
Decompiling & RE
Android OS and Security
Inspiring talks
APISecure conf workshop by Alissa Knight
Steps from the video
Step 1: Download the app in an android device and use apkextractor to extract it of of the android device
Step 2: Install MobSF
Inspect the results from the apk analysis here
Go to the Mobsf folder where everything as been extracted and make some manual checks.
You can use RegEXAPI with
grep -R _token
Resources mentioned in the video
Another video with Alissa Knight that is a great complement to the first one
How I hacked 30 mobile banking apps & the future of API Security, Alissa Knight, Aite Group - Apidays 2019
Last updated