diff --git a/Core2/Network/build.gradle.kts b/Core2/Network/build.gradle.kts new file mode 100644 index 000000000..67baddfd4 --- /dev/null +++ b/Core2/Network/build.gradle.kts @@ -0,0 +1,35 @@ +plugins { + alias(libs.plugins.android.library) + alias(libs.plugins.kotlin.android) +} + +android { + namespace = "com.infomaniak.network" + compileSdk = 34 + + defaultConfig { + minSdk = 24 + + testInstrumentationRunner = "androidx.test.runner.AndroidJUnitRunner" + consumerProguardFiles("consumer-rules.pro") + } + + buildTypes { + release { + isMinifyEnabled = false + proguardFiles(getDefaultProguardFile("proguard-android-optimize.txt"), "proguard-rules.pro") + } + } + compileOptions { + sourceCompatibility = JavaVersion.VERSION_1_8 + targetCompatibility = JavaVersion.VERSION_1_8 + } + kotlinOptions { + jvmTarget = "1.8" + } +} + +dependencies { + implementation(libs.androidx.core.ktx) + implementation(core2.sentry.android) +} diff --git a/Core2/Network/consumer-rules.pro b/Core2/Network/consumer-rules.pro new file mode 100644 index 000000000..e69de29bb diff --git a/Core2/Network/proguard-rules.pro b/Core2/Network/proguard-rules.pro new file mode 100644 index 000000000..481bb4348 --- /dev/null +++ b/Core2/Network/proguard-rules.pro @@ -0,0 +1,21 @@ +# Add project specific ProGuard rules here. +# You can control the set of applied configuration files using the +# proguardFiles setting in build.gradle. +# +# For more details, see +# http://developer.android.com/guide/developing/tools/proguard.html + +# If your project uses WebView with JS, uncomment the following +# and specify the fully qualified class name to the JavaScript interface +# class: +#-keepclassmembers class fqcn.of.javascript.interface.for.webview { +# public *; +#} + +# Uncomment this to preserve the line number information for +# debugging stack traces. +#-keepattributes SourceFile,LineNumberTable + +# If you keep the line number information, uncomment this to +# hide the original source file name. +#-renamesourcefileattribute SourceFile \ No newline at end of file diff --git a/Core2/Network/src/main/AndroidManifest.xml b/Core2/Network/src/main/AndroidManifest.xml new file mode 100644 index 000000000..15aa53ac3 --- /dev/null +++ b/Core2/Network/src/main/AndroidManifest.xml @@ -0,0 +1,22 @@ + + + + + + diff --git a/Core2/Network/src/main/java/com/infomaniak/network/NetworkAvailability.kt b/Core2/Network/src/main/java/com/infomaniak/network/NetworkAvailability.kt new file mode 100644 index 000000000..53ed43cf6 --- /dev/null +++ b/Core2/Network/src/main/java/com/infomaniak/network/NetworkAvailability.kt @@ -0,0 +1,127 @@ +/* + * Infomaniak Core - Android + * Copyright (C) 2024 Infomaniak Network SA + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ +package com.infomaniak.network + +import android.content.Context +import android.net.ConnectivityManager +import android.net.ConnectivityManager.NetworkCallback +import android.net.Network +import android.net.NetworkCapabilities +import android.net.NetworkRequest +import android.os.Build +import io.sentry.Sentry +import kotlinx.coroutines.CoroutineDispatcher +import kotlinx.coroutines.Dispatchers +import kotlinx.coroutines.channels.ProducerScope +import kotlinx.coroutines.channels.awaitClose +import kotlinx.coroutines.flow.Flow +import kotlinx.coroutines.flow.callbackFlow +import kotlinx.coroutines.launch +import kotlinx.coroutines.sync.Mutex +import kotlinx.coroutines.sync.withLock + +class NetworkAvailability(private val context: Context, private val ioDispatcher: CoroutineDispatcher = Dispatchers.IO) { + + private val connectivityManager by lazy { context.getSystemService(Context.CONNECTIVITY_SERVICE) as ConnectivityManager } + private val mutex = Mutex() + + val isNetworkAvailable: Flow = callbackFlow { + val networks = mutableListOf() + + val callback = object : NetworkCallback() { + + override fun onAvailable(network: Network) { + launch { + mutex.withLock { + networks.add(network) + sendNetworkAvailability(networks) + } + } + } + + override fun onLost(network: Network) { + launch { + mutex.withLock { + networks.remove(network) + sendNetworkAvailability(networks) + } + } + } + } + + launch(ioDispatcher) { + send(getInitialNetworkAvailability(connectivityManager)) + } + + registerNetworkCallback(connectivityManager, callback) + + awaitClose { unregisterNetworkCallback(connectivityManager, callback) } + } + + private suspend fun ProducerScope.registerNetworkCallback( + connectivityManager: ConnectivityManager, + callback: NetworkCallback, + ) { + runCatching { + connectivityManager.registerNetworkCallback(networkRequestBuilder(), callback) + }.onFailure { exception -> + // Fix potential Exception thrown by ConnectivityManager on Android 11 + // Already fixed in Android S and above + // https://issuetracker.google.com/issues/175055271 + Sentry.captureException(exception) + send(false) + } + } + + private fun unregisterNetworkCallback(connectivityManager: ConnectivityManager, callback: NetworkCallback) { + runCatching { + connectivityManager.unregisterNetworkCallback(callback) + }.onFailure { exception -> + Sentry.captureException(exception) + } + } + + @Suppress("DEPRECATION") + private fun getInitialNetworkAvailability(connectivityManager: ConnectivityManager): Boolean { + return if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) { + connectivityManager.activeNetwork?.let(::hasInternetConnectivity) ?: false + } else { + connectivityManager.activeNetworkInfo?.isConnected ?: false + } + } + + private fun ProducerScope.sendNetworkAvailability(networks: List) { + launch(ioDispatcher) { send(hasAvailableNetwork(networks)) } + } + + private fun networkRequestBuilder(): NetworkRequest { + return NetworkRequest.Builder().apply { + addCapability(NetworkCapabilities.NET_CAPABILITY_INTERNET) + }.build() + } + + private fun hasInternetConnectivity(network: Network) = runCatching { + network.getByName(ROOT_SERVER_URL) != null + }.getOrDefault(false) + + private suspend fun hasAvailableNetwork(networks: List) = mutex.withLock { networks.any(::hasInternetConnectivity) } + + companion object { + private const val ROOT_SERVER_URL = "a.root-servers.net" + } +} diff --git a/app/build.gradle.kts b/app/build.gradle.kts index 4f20591e2..112f8c80e 100644 --- a/app/build.gradle.kts +++ b/app/build.gradle.kts @@ -84,6 +84,7 @@ dependencies { implementation(project(":Core2")) implementation(project(":Core2:Sentry")) implementation(project(":Core2:Matomo")) + implementation(project(":Core2:Network")) implementation(project(":FileTypes")) implementation(kotlin("reflect")) diff --git a/settings.gradle.kts b/settings.gradle.kts index 57b8ebe0e..5e9ac91e7 100644 --- a/settings.gradle.kts +++ b/settings.gradle.kts @@ -32,4 +32,5 @@ include(":app") include(":Core2") include(":Core2:Sentry") include(":Core2:Matomo") +include(":Core2:Network") include(":FileTypes")