Common Android app vulnerabilities
Sebastian Porst (sporst@google.com) - LevelUp 0x05, October 2019
Introduction
How Google Play Protect educates developers to make
millions of apps safer to use
Talk structure
● GPSRP has handled 19 vulnerability classes in 6 different
categories
○ Walking through explanation, audit, and remediation for these takes
about 90 minutes.
○ For this talk I picked 6 representative vulnerabilities.
● Twice as much content is at https://hackerone.com/googleplay
Google Play Protect programs to help developers
ASI
● App Security Improvement Program
● Scans all apps on Google Play for security vulnerabilities and alerts developers
● Helped more than 300,000 developers fix more than 1,000,000 apps
GPSRP
● Google Play Security Rewards Program
● Bug bounty program collaboration between developers and Google
● Paid out more than $550,000 in 24 months
○ $362,300 in the last 90 days
Websites to help developers
From Google
● App security best practices: https://developer.android.com/topic/security/best-practices
● Don’t miss the other topics in /security/
From third-party websites
● Common Weakness Enumeration: https://cwe.mitre.org/data/definitions/919.html
● OWASP Mobile Security Project: https://www.owasp.org/index.php/OWASP_Mobile_Security_Project
Common Android app vulnerabilities
As proven by GPSRP payouts
Insecure connections
Insecure connections
● Use of insecure network protocols (CWE-319, $500 reward)
○ Loading data over insecure networks such as HTTP allows attackers that control the
network (for example a local WiFi network) to replace, remove, and inject code.
● Data transmission over insecure protocols (CWE-319, $3000 reward)
○ Sending sensitive data over insecure networks such as HTTP allows attackers that
control the network (for example a local WiFi network) to intercept, modify, and steal
the data.
● Authentication over insecure protocols (CWE-319, $3000 reward)
○ Sending login information over insecure networks such as HTTP allows attackers that
control the network (for example a local WiFi network) to intercept, modify, and steal
the data.
Use of insecure network protocols
The problem: Loading data over insecure networks such as HTTP allows attackers that control the
network (for example a local WiFi network) to replace, remove, and inject code.
Example code
URL url = new URL("http://www.google.com");
HttpURLConnection urlConnection =
(HttpURLConnection) url.openConnection();
urlConnection.connect();
InputStream in = urlConnection.getInputStream();
Use of insecure network protocols
The problem: Loading data over insecure networks such as HTTP allows attackers that control the
network (for example a local WiFi network) to replace, remove, and inject code.
Auditing tips
● Review the app’s network security configuration (an Android P feature) to understand
whether app is allowed to use insecure network protocols in the first place.
○ https://developer.android.com/training/articles/security-config (“Network
security configuration”)
● Grep the app’s code for “http:”, “ftp”, “smtp:”, and URLs that indicate use of insecure
network protocols.
● Understand common entry points into the network such as the URL class or the
WebView class and check whether there are code flows that lead to insecure
connections being made.
Use of insecure network protocols
The problem: Loading data over insecure networks such as HTTP allows attackers that control the
network (for example a local WiFi network) to replace, remove, and inject code.
Remediation
● Apps targeting Android P and above are safe by default. If such apps still want to
make insecure connections, they have to define a network security configuration
that allows that.
● Use javax.net.ssl.HttpsURLConnection as much as possible.
● Presubmit checks to look for URLs for insecure protocols are easy to set up.
● The Android Lint check InsecureBaseConfiguration flags insecure HTTP
connections.
Cryptography and authentication
Cryptography and authentication
● Embedded third-party secrets (CWE-798, $3000 reward)
○ Apps that embed third-party secrets such as Twitter API keys or AWS authentication
tokens can trivially have these secrets extracted and abused by attackers.
● Embedded cryptography secrets (CWE-321, $3000 reward)
○ Applications that use embedded crypto secrets are susceptible to simple data
decryption attacks.
● Leaking OAuth tokens (CWE-???, $3000 reward)
○ OAuth Implicit Grant and Authz (without PKCE) flows expose tokens
that can be used to create fraudulent requests.
Embedded third-party secrets
The problem: Apps that embed third-party secrets such as Twitter API keys or AWS authentication
tokens can trivially have these secrets extracted and abused by attackers.
Example code
ParseTwitterUtils.initialize(
"CONSUMER KEY", "CONSUMER SECRET");
Embedded third-party secrets
The problem: Apps that embed third-party secrets such as Twitter API keys or AWS authentication
tokens can trivially have these secrets extracted and abused by attackers.
Auditing tips
● Compile a list of interesting public APIs and create regular expressions to find keys or
secrets through grep.
● Compile a list of interesting public APIs and look for their package names in apps.
● Grep for “key”, “password”, “login”, “secret” and such.
Embedded third-party secrets
The problem: Apps that embed third-party secrets such as Twitter API keys or AWS authentication
tokens can trivially have these secrets extracted and abused by attackers.
Remediation
● Follow best practices for the used API as documented in their help center.
○ Some APIs encourage you to embed their secrets in the app as a secret
compromise will have limited negative effects.
● Some third-party services offer more secure alternatives to embedding credentials
in the app (such as Amazon and Google) while others like Twitter consider
embedding the third-party secrets in the code as best practice.
○ For services which don’t provide a per-user authentication service, it’s possible
to spin up your own server that handles this.
● For services that recommend embedding third-party secrets in the app, expect the
secrets to be extracted and abused by hackers. The goal for such situations is to
mitigate the negative effects of the abuse.
Private file access
Private file access
● Private data sharing (CWE-359, $3000 reward)
○ Applications may leak private data to other apps or attackers in obvious and subtle
ways.
● Private data overwrite due to path traversal (CWE category 21, $3000)
○ Applications that don’t sanitize attacker-provided directory paths may be susceptible
to overwrite their internal files with attacker-provided files.
● Private data overwrite due to ZIP file traversal (CWE-???, $3000
reward)
○ Applications that unzip archive files without sanitizing the target file paths of files inside
the archives are susceptible to overwriting their internal files with attacker-provided
files.
Private data overwrite due to ZIP file traversal
The problem: Applications that unzip archive files without sanitizing the target file paths of files
inside the archives are susceptible to overwriting their internal files with attacker-provided files.
Example code
InputStream is = new FileInputStream(zipFilePath);
ZipInputStream zis = new ZipInputStream(new BufferedInputStream(is));
ZipEntry ze;
while ((ze = zis.getNextEntry()) != null) {
outputFileName = ze.getName();
writeToFile(outputFileName, zis);
}
Private data overwrite due to ZIP file traversal
The problem: Applications that unzip archive files without sanitizing the target file paths of files
inside the archives are susceptible to overwriting their internal files with attacker-provided files.
Auditing tips
● Find all uses of APIs in the java.util.zip.* package or third-party zipping libraries.
● Understand whether user-provided ZIP files are being unzipped by the code.
Private data overwrite due to ZIP file traversal
The problem: Applications that unzip archive files without sanitizing the target file paths of files
inside the archives are susceptible to overwriting their internal files with attacker-provided files.
Remediation
● Validate that the canonical path of unzipped files points to the real directory to unzip
to.
○ A Google search for “android zip path traversal” will lead to examples.
Unprotected app parts
Unprotected app parts
● Unprotected activities (CWE-926, $3000 reward)
○ Exported activities can be started by other apps on the device which may break user
workflow assumptions, including ways that break security boundaries.
● Unprotected services (CWE-926, $3000 reward)
○ Applications that export services allow malicious apps on the device to start those
services.
● Typos in custom permissions (CWE-???, $3000 reward)
○ When used custom permissions don’t match declared custom permissions, Android
defaults to silently failing to enforce the permission.
Unprotected app parts
● Intent redirection (CWE-925, $500 reward)
○ Apps that accept and launch arbitrary intents from external sources may allow malware
to start internal components indirectly or access protected content:// URIs.
● Implicit broadcasts (sending) (CWE-927, $3000 reward)
○ Applications that send broadcasts without specifying the broadcast target may have
these broadcasts intercepted by malicious apps on the same device.
● Implicit broadcasts (receiving) (CWE-925, $3000 reward)
○ Applications that accept broadcasts without checking the sender may accept
maliciously crafted broadcasts sent from malicious apps on the same device.
Intent redirection
The problem: Apps that accept and launch arbitrary intents from external sources may allow
malware to start internal components indirectly or access protected content:// URIs.
Example code
protected void onHandleIntent(Intent intent) {
if (intent != null) {
Intent privateIntent = new Intent(this, parseIntentClass(intent));
startActivity(privateIntent);
}
}
Intent redirection
The problem: Apps that accept and launch arbitrary intents from external sources may allow
malware to start internal components indirectly or access protected content:// URIs.
Auditing tips
● Find calls to startActivity and verify that Intent components are constructed from
trusted data.
● Find calls Intent::getExtras where returned values are cast to Intent and verify that
they are properly checked before being used.
○ It isn’t enough to check the target class name. Malware can reuse your class
names and force your app to send an Intent that grants content:// URI access to
their app.
Intent redirection
The problem: Apps that accept and launch arbitrary intents from external sources may allow
malware to start internal components indirectly or access protected content:// URIs.
Remediation
● Whenever sending an Intent created from untrusted data
○ Check the class name and package name of the target component or…
○ Check the signing key of the app that owns the target component
Other
Other
● Incorrect URL verification (CWE-939, $500 reward)
○ Apps that rely on URL parsing to verify that a given URL is pointing to a trusted server
may be susceptible to many different ways to get URL parsing and verification wrong.
● Cross-app scripting (CWE-???, $3000 reward)
○ Apps that load untrusted URLs from other apps into their WebViews that match a
certain form (e.g., javascript: or file:///path/to/private) allow malicious JavaScript code
execution.
● Incorrect sandboxing of scripting language (CWE-266, $20,000 reward)
○ Embedding a scripting language or interpreter in an app can lead to exposure of app
internals if the security boundaries of the interpreter are not well understood.
● Unprotected data on server (CWE-306, $3000 reward)
○ An app connects to a remote web server, database, or API that does not sufficiently
protect sensitive user data. Any hacker can access this data.
Incorrect URL verification
The problem: Apps that rely on URL parsing to verify that a given URL is pointing to a trusted server
may be susceptible to many different ways to get URL parsing and verification wrong.
Example code
// Just one of many ways to get URL verification wrong
if (uri.getHost().endswith("mywebsite.com")) {
webView.loadUrl(uri.toString());
}
Incorrect URL verification
The problem: Apps that rely on URL parsing to verify that a given URL is pointing to a trusted server
may be susceptible to many different ways to get URL parsing and verification wrong.
Auditing tips
● Find every instance of URLs being used in an app
● Narrow it down to those that work on user-controlled input
● Narrow it down to those that are used to branch between different code paths
○ Could be any part of the URL, not just the host name
Incorrect URL verification
The problem: Apps that rely on URL parsing to verify that a given URL is pointing to a trusted server
may be susceptible to many different ways to get URL parsing and verification wrong.
Remediation
● Read and understand https://hackerone.com/reports/431002
○ “Golden techniques to bypass host validations in Android apps”
● Create a library that follows best practices and always use that when trying to verify
URLs
Bonus if we have more time
Incorrect sandboxing of scripting language
The problem: Embedding a scripting language or interpreter in an app can lead to exposure of app
internals if the security boundaries of the interpreter are not well understood.
Example code
PythonInterpreter interpreter = new PythonInterpreter();
interpreter.exec(
"from shutil import copyfile\n
copyfile(\"passwords.bin\", \"/sdcard/passwords.bin\"
);
Incorrect sandboxing of scripting language
The problem: Embedding a scripting language or interpreter in an app can lead to exposure of app
internals if the security boundaries of the interpreter are not well understood.
Auditing tips
● Look for indicators of scripting languages
○ Common interpreters such as Python inside the app.
○ Interpreters may be hidden in file formats like PDF (Javascript).
○ Look for Android and Java sandboxing libraries.
● Check whether these interpreters allow untrusted input.
Incorrect sandboxing of scripting language
The problem: Embedding a scripting language or interpreter in an app can lead to exposure of app
internals if the security boundaries of the interpreter are not well understood.
Remediation
● Disable user-supplied input if possible.
● Build your own limited-scope interpreter or interface to more permissive
interpreters.
● Otherwise sanitize scripts, disable scripting, or restrict permissions and abilities of
interpreter engine.
THANK
YOU