Fortifying Android Apps: A Comprehensive Guide to Securing Your App and Codebase with Kotlin

Syed Ovais Akhtar
11 min readDec 18, 2023

--

In an era dominated by mobile technology, Android applications have become an integral part of our daily lives, serving as gateways to information, communication, and entertainment. As the demand for feature-rich and user-friendly apps continues to soar, developers face the critical challenge of not only delivering innovative functionalities but also ensuring the robust security of their creations. In this dynamic landscape, where cyber threats are evolving at an unprecedented pace, safeguarding sensitive user data has never been more crucial.

This article aims to shed light on the imperative task of securing Android applications, leveraging the power of Kotlin — the modern, concise, and expressive programming language. Kotlin has rapidly gained popularity among Android developers for its seamless interoperability with Java, conciseness, and null safety features. By delving into the key security considerations and best practices, developers can fortify their Android applications against a myriad of potential vulnerabilities, ensuring that users can interact with their apps confidently, knowing that their data is shielded from malicious actors.

Join us on a journey through the realms of secure Android app development, where we will explore the common security pitfalls, the importance of encryption, secure data storage, and authentication mechanisms. By the end of this guide, you’ll be armed with the knowledge and tools necessary to navigate the intricate landscape of Android app security, providing your users with a shielded and trustworthy digital experience. Let’s embark on the quest to empower your Android applications with the robust security measures they deserve, ensuring a resilient defense against the ever-evolving threat landscape.

We broke-down our strategies into sub components

Let’s take a look!

Validate App Cloning

App cloning in Android refers to the process of creating a duplicate or copy of an existing mobile application. This practice has both legitimate and illegitimate use cases. Legitimate uses may include creating separate accounts for a single app or testing different versions of an application simultaneously. However, it’s crucial to note that app cloning can also be exploited for malicious purposes, such as creating fake or fraudulent versions of popular apps to deceive users.

Let’s deep dive into it’s implementation

We are validating our package name first and then we are validating that our app is installed from PlayStore or Not.

Note: PACKAGE_NAME is our app’s package name which we described when creating our project or is visible in build.gradle(app-level).

Validating Rooted Device

A rooted Android device refers to a device where the user has gained root access or administrative privileges, allowing them to modify the system files and settings beyond what is typically permitted by the device manufacturer. Rooting provides users with elevated privileges, giving them more control over the device’s operating system. While rooting can offer several advantages, it also comes with certain risks and considerations.

Validating Virtual Machines

Running a virtual machine (VM) on an Android device involves using virtualization technology to create an isolated environment within which a separate operating system can run. While running full-fledged VMs on Android may be challenging due to hardware and resource limitations, there are some applications and solutions that allow for a level of virtualization on Android devices. Here are a few methods and considerations:

Emulators:

Android itself is often run in an emulated environment on other platforms, but there are also emulators available for Android that let you run different operating systems. An example is QEMU (Quick EMUlator), which allows you to emulate various architectures and run different operating systems on your Android device.

Termux and Linux Deploy:

Apps like Termux and Linux Deploy on the Google Play Store enable you to install a Linux distribution (such as Debian or Ubuntu) within a chroot environment. While not a full VM, this provides a way to run Linux commands and applications on your Android device.

Virtual Machine Apps:

Some apps on the Play Store claim to provide virtualization capabilities. These apps typically create a virtual environment within which you can install and run a guest operating system. However, these solutions might have limitations in terms of supported operating systems and performance.

Android-x86:

Android-x86 is an open-source project that ports Android to the x86 architecture. You can install Android-x86 on a virtual machine using software like VirtualBox or VMware. While this is not running a VM on Android directly, it allows you to run Android in a virtualized environment on a different platform.

AnLinux and UserLAnd:

Apps like AnLinux and UserLAnd allow you to run Linux distributions alongside your Android system. These apps often use chroot environments to provide a Linux shell and applications.

It’s important to note that running VMs on Android devices may have performance limitations, and not all devices support virtualization technology. Additionally, the level of compatibility and functionality can vary depending on the specific solution you choose.

Before attempting to run a virtual machine on your Android device, ensure that your device supports virtualization, and be aware of any potential risks or limitations associated with the chosen method. Always follow the instructions provided by the developers of the specific app or solution you are using.

Let’s see how we’re checking virtual machine in our app.

Validating Debuggable Build

We are verifying that our production release is not debuggable

Validating Debugger

We are validating that debugger isn’t attached to our release.

Validating Thread CPU Time

Validating thread CPU time in the context of Android app security involves monitoring and assessing the CPU usage of threads within an application. This process helps identify potential performance issues, resource bottlenecks, or even malicious activities that may impact the overall security and stability of the app.

Securing Shared Preferences

Encrypted Shared Preferences in Android provide a secure way to store sensitive information such as user credentials, API keys, or other confidential data within your application. The standard SharedPreferences in Android stores data in an unencrypted form, making it susceptible to unauthorized access. By implementing encryption, you add an additional layer of security to protect sensitive information from potential threats.

Let’s deep dive into the implementation

Let’s see how it look like.

Device Explorer -> data -> data -> your_package_name -> shared_prefes -> your_preference_filename

Our keys and values both are encrypted here.

Usability

Let’s see our final validation file

System Property Provider

Code Obfuscation

Code obfuscation is a technique used in software development, including Android app development, to make the source code more difficult to understand or reverse engineer. The primary goal of code obfuscation is to enhance the security of an application by making it challenging for attackers to analyze and decompile the code. ProGuard is a popular tool for code obfuscation in Android development.

  1. Enable proguard from build.gradle
  2. Create proguard rule

Let’s see an example

Let’s what are dontwarn, keep, dontnote,keepattributes etc. are.

DontWarn:

In ProGuard, the -dontwarn directive is used to suppress warning messages that may be generated during the code obfuscation and optimization process. ProGuard analyzes and processes the code, and in some cases, it may encounter situations where it issues warning messages. The -dontwarn directive allows you to instruct ProGuard to ignore specific warning messages related to classes, methods, or fields.

Keep:

In ProGuard, the -keep directive is used to specify classes, methods, and fields that should not be removed or obfuscated during the code shrinking and obfuscation process. The -keep directive is essential for preserving specific parts of your code that may otherwise be stripped away or renamed by ProGuard. It helps ensure that certain classes, methods, or fields retain their original names and functionality.

DontNote:

In ProGuard, the -dontnote directive is used to suppress informational notes that may be generated during the code processing phase. These notes are not warnings or errors, but rather informative messages that ProGuard provides during its analysis and optimization processes. The -dontnote directive allows you to instruct ProGuard to ignore specific informational notes, which can help reduce the amount of output generated during the build process.

KeepAttributes:

In ProGuard, the -keepattributes directive is used to specify which attributes of classes, methods, or fields should be preserved during the obfuscation process. An attribute in this context refers to additional metadata associated with classes, methods, or fields, and ProGuard allows you to selectively retain or discard certain attributes.

Token Based Authentication

Token-based authentication is a common approach to secure access to resources in a mobile application, including Android apps. It involves the use of tokens, which are short-lived, unique pieces of information that serve as credentials to authenticate users and authorize their access to protected resources. The two main types of tokens used in token-based authentication are access tokens and refresh tokens.

Here’s a high-level overview of how token-based authentication works in the context of an Android application:

User Authentication:

When a user logs into the Android app, their credentials (username and password) are sent securely to an authentication server.

Token Issuance:

If the credentials are valid, the authentication server generates two types of tokens:

  • Access Token: A short-lived token that represents the user’s identity and contains information about their permissions.
  • Refresh Token: A longer-lived token used to obtain a new access token when the current one expires.

Token Storage on the Device:

The Android app stores the access token and refresh token securely on the device. Common storage options include SharedPreferences, encrypted databases, or secure storage solutions.

Token Usage for API Requests:

When the app needs to access protected resources (e.g., making API requests), it includes the access token in the request headers. The server verifies the token to ensure the user has the necessary permissions.

Token Expiration and Refresh:

Access tokens have a limited lifespan to enhance security. When an access token expires, the app uses the refresh token to request a new access token without requiring the user to log in again.

Token Revocation:

If a user logs out or if there are security concerns, the app can revoke the tokens by notifying the authentication server. Revoked tokens are no longer valid for authentication.

Benefits of Token-Based Authentication in Android:

  • Statelessness: Tokens are self-contained, meaning the server does not need to store information about the user’s session. This makes the authentication process more scalable.
  • Security: Tokens are often transmitted over secure channels (e.g., HTTPS) and can be encrypted to protect sensitive information. Regularly refreshing access tokens enhances security.
  • Cross-Origin Resource Sharing (CORS): Tokens can be used to authenticate requests from different origins, facilitating the development of mobile apps that interact with various APIs.
  • Decoupling: Token-based authentication decouples the authentication server from the resource server, allowing different services to be used independently.

When implementing token-based authentication in Android, developers commonly use OAuth 2.0 or JSON Web Tokens (JWT) as the underlying mechanisms for issuing and verifying tokens. Libraries and frameworks are available to simplify the integration of token-based authentication into Android applications, making it a popular choice for securing mobile app interactions with APIs and services.

Secure Communication

Secure communication refers to network communication that is transmitted over an unencrypted connection, leaving the data vulnerable to interception and unauthorized access. It is crucial to secure network traffic, especially when dealing with sensitive information, such as user credentials, personal data, or confidential transactions. Android provides mechanisms to enforce secure communication, and developers are encouraged to avoid transmitting sensitive data in clear text.

Here are some key considerations and practices to ensure secure network communication in Android:

Use HTTPS (SSL/TLS)

Always use HTTPS (SSL/TLS) for transmitting sensitive data over the network. This ensures that the communication is encrypted, preventing eavesdropping and man-in-the-middle attacks.

Network Security Configuration:

Implement network security configuration to specify the security policies for your app’s network communication. This includes defining a trust manager, configuring cleartext traffic policies, and specifying secure connections.

values/xml/config.xml

Enforce Cleartext Traffic Restriction:

AndroidManifest.xml

Avoid Allow-All Trust Managers:

When using a custom trust manager for secure connections, avoid using trust managers that accept any certificate without validation. This can expose your app to security risks.

Implement Certificate Pinning:

Consider implementing certificate pinning to enhance the security of your app. This ensures that your app only communicates with servers whose certificates match predefined values.

Conclusion

In conclusion, fortifying your Android applications with Kotlin is not merely a prudent practice; it is an essential imperative in an era where digital threats loom large. The journey through the intricacies of secure Android app development using Kotlin has unveiled a plethora of robust techniques and best practices. From encryption and secure data storage to vigilant authentication mechanisms, the arsenal of tools available to developers is both comprehensive and dynamic.

As we navigate the ever-evolving landscape of mobile technology, it becomes evident that user trust and data integrity are non-negotiable pillars of app development. By embracing the power of Kotlin, developers empower themselves to not only craft innovative and efficient applications but also to erect formidable defenses against cyber threats.

The importance of proactive security measures cannot be overstated. As developers, it is our responsibility to stay abreast of emerging security trends, continuously refine our coding practices, and remain committed to the safeguarding of user information. The synergy between Kotlin’s expressive syntax and its interoperability with Java positions it as a formidable ally in the quest for secure Android app development.

In the final analysis, the adoption of security-conscious coding practices is not a one-time endeavor but a perpetual commitment to the integrity of our digital creations. As we usher our Android applications into an era of heightened connectivity and interactivity, let the principles outlined in this guide serve as a steadfast guidepost on the journey toward crafting not only functional but fortified applications that stand resilient in the face of evolving security challenges. Through the judicious application of Kotlin and a steadfast dedication to security, we pave the way for a future where user confidence and app resilience coalesce, ensuring a safer and more trustworthy digital landscape for all.

For more updates Follow me on Medium, GitHub & LinkedIn.

--

--

Syed Ovais Akhtar
Syed Ovais Akhtar

Written by Syed Ovais Akhtar

Senior Software Engineer - Mobile

No responses yet