Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Adds App Store payment support on Calendar App #8043

Merged
merged 6 commits into from
Dec 9, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -6,13 +6,16 @@ import android.content.ClipData
import android.content.Intent
import android.net.Uri
import android.provider.Settings
import android.util.Base64
import android.util.Log
import androidx.biometric.BiometricManager
import androidx.biometric.BiometricPrompt
import androidx.core.app.ActivityCompat.startActivityForResult
import androidx.core.content.ContextCompat.startActivity
import androidx.core.content.FileProvider
import androidx.fragment.app.FragmentActivity
import de.tutao.tutashared.CredentialAuthenticationException
import de.tutao.tutashared.SystemUtils
import de.tutao.tutashared.atLeastTiramisu
import de.tutao.tutashared.credentials.AuthenticationPrompt
import de.tutao.tutashared.data.AppDatabase
Expand All @@ -21,7 +24,7 @@ import kotlinx.coroutines.Dispatchers
import kotlinx.coroutines.withContext
import java.io.File
import java.io.IOException
import de.tutao.tutashared.SystemUtils
import java.nio.charset.Charset


class AndroidMobileSystemFacade(
Expand All @@ -35,6 +38,8 @@ class AndroidMobileSystemFacade(
companion object {
private const val TAG = "SystemFacade"
const val APP_LOCK_METHOD = "AppLockMethod"
const val TUTA_INTENT_ACTION = "TUTA_INTEROP"
const val TUTA_INTENT_INTEROP_DATA = "TUTA_INTEROP_DATA"
}

override suspend fun openLink(uri: String): Boolean {
Expand Down Expand Up @@ -171,6 +176,37 @@ class AndroidMobileSystemFacade(
Log.e(TAG, "Trying to open Tuta Mail from Tuta Mail")
}

private fun tryToLaunchStore() {
try {
val packageId = activity.getString(R.string.package_name).replace("tutanota", "calendar")
startActivity(
activity,
Intent(Intent.ACTION_VIEW, Uri.parse("market://details?id=$packageId")),
null
)
} catch (e: Exception) {
Log.d(TAG, "Failed to launch store $e")
}
}

override suspend fun openCalendarApp(query: String) {
val decodedQuery = Base64.decode(query.toByteArray(), Base64.DEFAULT).toString(Charset.defaultCharset())
val targetPackageId = activity.getString(R.string.package_name).replace("calendar", "tutanota")

val intent = Intent()
intent.setPackage(targetPackageId)
intent.setAction(Intent.ACTION_EDIT)
intent.putExtra(TUTA_INTENT_ACTION, "interop")
intent.setData(Uri.parse("tutacalendar://interop?$decodedQuery"))

try {
startActivityForResult(activity, intent, 0, null)
} catch (e: Exception) {
Log.d(TAG, e.toString())
tryToLaunchStore()
}
}

override suspend fun getInstallationDate(): String {
return SystemUtils.getInstallationDate(activity.packageManager, activity.packageName)
}
Expand Down
11 changes: 10 additions & 1 deletion app-android/app/src/main/java/de/tutao/tutanota/MainActivity.kt
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ import android.content.ClipData
import android.content.ClipboardManager
import android.content.ComponentName
import android.content.Intent
import android.content.Intent.ACTION_EDIT
import android.content.pm.PackageManager
import android.content.res.Configuration
import android.graphics.Color
Expand Down Expand Up @@ -453,7 +454,13 @@ class MainActivity : FragmentActivity() {
return@launch
}

if (data != null && data.toString().startsWith("tutamail://") && data.host == "interop") {
val isInteropCall = intent.action == ACTION_EDIT && intent.getStringExtra(TUTA_INTENT_ACTION) == "interop"
val isTrustedCaller = callingPackage == BuildConfig.APPLICATION_ID.replace(
"tutanota",
"calendar"
)

if (data != null && isInteropCall && isTrustedCaller) {
openContactEditor(data)
}

Expand Down Expand Up @@ -804,6 +811,8 @@ class MainActivity : FragmentActivity() {
const val OPEN_CONTACT_EDITOR_CONTACT_ID = "contactId"
const val OPEN_USER_MAILBOX_MAILID_KEY = "mailId"
const val ALREADY_HANDLED_INTENT = "alreadyHandledIntent"
const val TUTA_INTENT_ACTION = "TUTA_INTEROP"

private const val TAG = "MainActivity"
private var requestId = 0

Expand Down
9 changes: 9 additions & 0 deletions app-android/calendar/src/main/AndroidManifest.xml
Original file line number Diff line number Diff line change
Expand Up @@ -65,6 +65,15 @@

<action android:name="android.intent.action.VIEW"/>
</intent-filter>
<intent-filter>
<data
android:scheme="tutacalendar"
android:host="interop"/>
<action android:name="android.intent.action.EDIT"/>

<category android:name="android.intent.category.DEFAULT"/>
<category android:name="android.intent.category.BROWSABLE"/>
</intent-filter>
<intent-filter>
<data
android:scheme="tutacalendar"
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@ import android.util.Base64
import android.util.Log
import androidx.biometric.BiometricManager
import androidx.biometric.BiometricPrompt
import androidx.core.app.ActivityCompat.startActivityForResult
import androidx.core.content.ContextCompat.startActivity
import androidx.core.content.FileProvider
import androidx.fragment.app.FragmentActivity
Expand All @@ -32,12 +33,13 @@ class AndroidMobileSystemFacade(
private val activity: MainActivity,
private val db: AppDatabase,
) : MobileSystemFacade {

private val authenticationPrompt = AuthenticationPrompt()

companion object {
private const val TAG = "SystemFacade"
const val APP_LOCK_METHOD = "AppLockMethod"
const val TUTA_INTENT_ACTION = "TUTA_INTEROP"
const val TUTA_INTENT_INTEROP_DATA = "TUTA_INTEROP_DATA"
}

override suspend fun openLink(uri: String): Boolean {
Expand Down Expand Up @@ -170,11 +172,12 @@ class AndroidMobileSystemFacade(
}
}

fun tryToLaunchStore() {
private fun tryToLaunchStore() {
try {
val packageId = activity.getString(R.string.package_name).replace("calendar", "tutanota")
startActivity(
activity,
Intent(Intent.ACTION_VIEW, Uri.parse("market://details?id=de.tutao.tutanota")),
Intent(Intent.ACTION_VIEW, Uri.parse("market://details?id=$packageId")),
null
)
} catch (e: Exception) {
Expand All @@ -184,19 +187,26 @@ class AndroidMobileSystemFacade(

override suspend fun openMailApp(query: String) {
val decodedQuery = Base64.decode(query.toByteArray(), Base64.DEFAULT).toString(Charset.defaultCharset())
val targetPackageId = activity.getString(R.string.package_name).replace("calendar", "tutanota")

val intent = Intent()
intent.setPackage(targetPackageId)
intent.setAction(Intent.ACTION_EDIT)
intent.setData(Uri.parse("tutamail://interop?${decodedQuery}"))
intent.putExtra(TUTA_INTENT_ACTION, "interop")
intent.setData(Uri.parse("tutamail://interop?$decodedQuery"))

try {
startActivity(activity, intent, null)
startActivityForResult(activity, intent, 0, null)
} catch (e: Exception) {
Log.d(TAG, e.toString())
tryToLaunchStore()
}
}

override suspend fun openCalendarApp(query: String) {
Log.e(TAG, "Trying to open Tuta Calendar from Tuta Calendar")
}

override suspend fun getInstallationDate(): String {
return SystemUtils.getInstallationDate(activity.packageManager, activity.packageName)
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -66,4 +66,10 @@ interface CommonNativeFacade {
suspend fun handleFileImport(
filesUris: List<String>,
): Unit
/**
* Open a specified path inside settings
*/
suspend fun openSettings(
path: String,
): Unit
}
Original file line number Diff line number Diff line change
Expand Up @@ -123,4 +123,14 @@ class CommonNativeFacadeSendDispatcher (
this.transport.sendRequest("ipc", listOf(encodedFacade, encodedMethod) + args)
}

override suspend fun openSettings(
path: String,
): Unit
{
val encodedMethod = json.encodeToString("openSettings")
val args : MutableList<String> = mutableListOf()
args.add(json.encodeToString(path))
this.transport.sendRequest("ipc", listOf(encodedFacade, encodedMethod) + args)
}

}
Original file line number Diff line number Diff line change
Expand Up @@ -53,6 +53,9 @@ interface MobileSystemFacade {
suspend fun openMailApp(
query: String,
): Unit
suspend fun openCalendarApp(
query: String,
): Unit
/**
* Returns the date and time the app was installed as a string with milliseconds in UNIX epoch.
*/
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,7 @@ class MobileSystemFacadeReceiveDispatcher(
private val json: Json,
private val facade: MobileSystemFacade,
) {

suspend fun dispatch(method: String, arg: List<String>): String {
when (method) {
"goToSettings" -> {
Expand Down Expand Up @@ -90,6 +90,13 @@ class MobileSystemFacadeReceiveDispatcher(
)
return json.encodeToString(result)
}
"openCalendarApp" -> {
val query: String = json.decodeFromString(arg[0])
val result: Unit = this.facade.openCalendarApp(
query,
)
return json.encodeToString(result)
}
else -> throw Error("unknown method for MobileSystemFacade: $method")
}
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -2,17 +2,16 @@


@file:Suppress("NAME_SHADOWING")

package de.tutao.tutashared.ipc

import kotlinx.serialization.encodeToString
import kotlinx.serialization.json.Json
import kotlinx.serialization.*
import kotlinx.serialization.json.*

class NativeCryptoFacadeReceiveDispatcher(
private val json: Json,
private val facade: NativeCryptoFacade,
) {

suspend fun dispatch(method: String, arg: List<String>): String {
when (method) {
"rsaEncrypt" -> {
Expand All @@ -26,7 +25,6 @@ class NativeCryptoFacadeReceiveDispatcher(
)
return json.encodeToString(result)
}

"rsaDecrypt" -> {
val privateKey: RsaPrivateKey = json.decodeFromString(arg[0])
val data: DataWrapper = json.decodeFromString(arg[1])
Expand All @@ -36,7 +34,6 @@ class NativeCryptoFacadeReceiveDispatcher(
)
return json.encodeToString(result)
}

"aesEncryptFile" -> {
val key: DataWrapper = json.decodeFromString(arg[0])
val fileUri: String = json.decodeFromString(arg[1])
Expand All @@ -48,7 +45,6 @@ class NativeCryptoFacadeReceiveDispatcher(
)
return json.encodeToString(result)
}

"aesDecryptFile" -> {
val key: DataWrapper = json.decodeFromString(arg[0])
val fileUri: String = json.decodeFromString(arg[1])
Expand All @@ -58,7 +54,6 @@ class NativeCryptoFacadeReceiveDispatcher(
)
return json.encodeToString(result)
}

"argon2idGeneratePassphraseKey" -> {
val passphrase: String = json.decodeFromString(arg[0])
val salt: DataWrapper = json.decodeFromString(arg[1])
Expand All @@ -68,15 +63,13 @@ class NativeCryptoFacadeReceiveDispatcher(
)
return json.encodeToString(result)
}

"generateKyberKeypair" -> {
val seed: DataWrapper = json.decodeFromString(arg[0])
val result: KyberKeyPair = this.facade.generateKyberKeypair(
seed,
)
return json.encodeToString(result)
}

"kyberEncapsulate" -> {
val publicKey: KyberPublicKey = json.decodeFromString(arg[0])
val seed: DataWrapper = json.decodeFromString(arg[1])
Expand All @@ -86,7 +79,6 @@ class NativeCryptoFacadeReceiveDispatcher(
)
return json.encodeToString(result)
}

"kyberDecapsulate" -> {
val privateKey: KyberPrivateKey = json.decodeFromString(arg[0])
val ciphertext: DataWrapper = json.decodeFromString(arg[1])
Expand All @@ -96,7 +88,6 @@ class NativeCryptoFacadeReceiveDispatcher(
)
return json.encodeToString(result)
}

else -> throw Error("unknown method for NativeCryptoFacade: $method")
}
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,8 @@

package de.tutao.tutashared.ipc

import kotlinx.serialization.Serializable
import kotlinx.serialization.*
import kotlinx.serialization.json.*


/**
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,8 @@

package de.tutao.tutashared.ipc

import kotlinx.serialization.Serializable
import kotlinx.serialization.*
import kotlinx.serialization.json.*


/**
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -63,4 +63,10 @@ public protocol CommonNativeFacade {
func handleFileImport(
_ filesUris: [String]
) async throws -> Void
/**
* Open a specified path inside settings
*/
func openSettings(
_ path: String
) async throws -> Void
}
Original file line number Diff line number Diff line change
Expand Up @@ -129,4 +129,15 @@ public class CommonNativeFacadeSendDispatcher : CommonNativeFacade {
let _ = try await self.transport.sendRequest(requestType: "ipc", args: [encodedFacadeName, encodedMethodName] + args)
}

public func openSettings(
_ path: String
) async throws -> Void
{
var args = [String]()
args.append(toJson(path))
let encodedFacadeName = toJson("CommonNativeFacade")
let encodedMethodName = toJson("openSettings")
let _ = try await self.transport.sendRequest(requestType: "ipc", args: [encodedFacadeName, encodedMethodName] + args)
}

}
Original file line number Diff line number Diff line change
Expand Up @@ -50,6 +50,9 @@ public protocol MobileSystemFacade {
func openMailApp(
_ query: String
) async throws -> Void
func openCalendarApp(
_ query: String
) async throws -> Void
/**
* Returns the date and time the app was installed as a string with milliseconds in UNIX epoch.
*/
Expand Down
Loading
Loading