mirror of
				https://github.com/recloudstream/cloudstream.git
				synced 2024-08-15 01:53:11 +00:00 
			
		
		
		
	First steps for multiplatform API (#1003)
* First steps for multiplatform api * Buildconfig testing * Fix publishing and classes.jar * Update build.gradle.kts
This commit is contained in:
		
							parent
							
								
									5db541d7cc
								
							
						
					
					
						commit
						6df3ef14f6
					
				
					 13 changed files with 185 additions and 41 deletions
				
			
		|  | @ -1,5 +1,6 @@ | |||
| import com.android.build.gradle.internal.cxx.configure.gradleLocalProperties | ||||
| import org.jetbrains.dokka.gradle.DokkaTask | ||||
| import org.jetbrains.kotlin.gradle.plugin.mpp.pm20.util.archivesName | ||||
| import org.jetbrains.kotlin.gradle.tasks.KotlinCompile | ||||
| import java.io.ByteArrayOutputStream | ||||
| import java.net.URL | ||||
|  | @ -13,6 +14,7 @@ plugins { | |||
| 
 | ||||
| val tmpFilePath = System.getProperty("user.home") + "/work/_temp/keystore/" | ||||
| val prereleaseStoreFile: File? = File(tmpFilePath).listFiles()?.first() | ||||
| var isLibraryDebug = false | ||||
| 
 | ||||
| fun String.execute() = ByteArrayOutputStream().use { baot -> | ||||
|     if (project.exec { | ||||
|  | @ -103,6 +105,7 @@ android { | |||
|             ) | ||||
|         } | ||||
|         debug { | ||||
|             isLibraryDebug = true | ||||
|             isDebuggable = true | ||||
|             applicationIdSuffix = ".debug" | ||||
|             proguardFiles( | ||||
|  | @ -232,18 +235,37 @@ dependencies { | |||
|     implementation("androidx.work:work-runtime:2.9.0") | ||||
|     implementation("androidx.work:work-runtime-ktx:2.9.0") | ||||
|     implementation("com.github.Blatzar:NiceHttp:0.4.11") // HTTP Lib | ||||
| 
 | ||||
|     implementation(project(":library") { | ||||
|         this.extra.set("isDebug", isLibraryDebug) | ||||
|     }) | ||||
| } | ||||
| 
 | ||||
| tasks.register("androidSourcesJar", Jar::class) { | ||||
| tasks.register<Jar>("androidSourcesJar") { | ||||
|     archiveClassifier.set("sources") | ||||
|     from(android.sourceSets.getByName("main").java.srcDirs) // Full Sources | ||||
| } | ||||
| 
 | ||||
| // For GradLew Plugin | ||||
| tasks.register("makeJar", Copy::class) { | ||||
|     from("build/intermediates/compile_app_classes_jar/prereleaseDebug") | ||||
|     into("build") | ||||
|     include("classes.jar") | ||||
| tasks.register<Copy>("copyJar") { | ||||
|     from( | ||||
|         "build/intermediates/compile_app_classes_jar/prereleaseDebug", | ||||
|         "../library/build/libs" | ||||
|     ) | ||||
|     into("build/app-classes") | ||||
|     include("classes.jar", "library-jvm*.jar") | ||||
|     // Remove the version | ||||
|     rename("library-jvm.*.jar", "library-jvm.jar") | ||||
| } | ||||
| 
 | ||||
| // Merge the app classes and the library classes into classes.jar | ||||
| tasks.register<Jar>("makeJar") { | ||||
|     dependsOn(tasks.getByName("copyJar")) | ||||
|     from( | ||||
|         zipTree("build/app-classes/classes.jar"), | ||||
|         zipTree("build/app-classes/library-jvm.jar") | ||||
|     ) | ||||
|     destinationDirectory.set(layout.buildDirectory) | ||||
|     archivesName = "classes" | ||||
| } | ||||
| 
 | ||||
| tasks.withType<KotlinCompile> { | ||||
|  |  | |||
|  | @ -743,8 +743,6 @@ fun base64Encode(array: ByteArray): String { | |||
|     } | ||||
| } | ||||
| 
 | ||||
| class ErrorLoadingException(message: String? = null) : Exception(message) | ||||
| 
 | ||||
| fun MainAPI.fixUrlNull(url: String?): String? { | ||||
|     if (url.isNullOrEmpty()) { | ||||
|         return null | ||||
|  |  | |||
|  | @ -1,211 +0,0 @@ | |||
| package com.lagradost.cloudstream3.mvvm | ||||
| 
 | ||||
| import android.util.Log | ||||
| import androidx.lifecycle.LifecycleOwner | ||||
| import androidx.lifecycle.LiveData | ||||
| import com.bumptech.glide.load.HttpException | ||||
| import com.lagradost.cloudstream3.BuildConfig | ||||
| import com.lagradost.cloudstream3.ErrorLoadingException | ||||
| import kotlinx.coroutines.* | ||||
| import java.io.InterruptedIOException | ||||
| import java.net.SocketTimeoutException | ||||
| import java.net.UnknownHostException | ||||
| import javax.net.ssl.SSLHandshakeException | ||||
| import kotlin.coroutines.CoroutineContext | ||||
| import kotlin.coroutines.EmptyCoroutineContext | ||||
| 
 | ||||
| const val DEBUG_EXCEPTION = "THIS IS A DEBUG EXCEPTION!" | ||||
| const val DEBUG_PRINT = "DEBUG PRINT" | ||||
| 
 | ||||
| class DebugException(message: String) : Exception("$DEBUG_EXCEPTION\n$message") | ||||
| 
 | ||||
| inline fun debugException(message: () -> String) { | ||||
|     if (BuildConfig.DEBUG) { | ||||
|         throw DebugException(message.invoke()) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| inline fun debugPrint(tag: String = DEBUG_PRINT, message: () -> String) { | ||||
|     if (BuildConfig.DEBUG) { | ||||
|         Log.d(tag, message.invoke()) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| inline fun debugWarning(message: () -> String) { | ||||
|     if (BuildConfig.DEBUG) { | ||||
|         logError(DebugException(message.invoke())) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| inline fun debugAssert(assert: () -> Boolean, message: () -> String) { | ||||
|     if (BuildConfig.DEBUG && assert.invoke()) { | ||||
|         throw DebugException(message.invoke()) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| inline fun debugWarning(assert: () -> Boolean, message: () -> String) { | ||||
|     if (BuildConfig.DEBUG && assert.invoke()) { | ||||
|         logError(DebugException(message.invoke())) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| /** NOTE: Only one observer at a time per value */ | ||||
| fun <T> LifecycleOwner.observe(liveData: LiveData<T>, action: (t: T) -> Unit) { | ||||
|     liveData.removeObservers(this) | ||||
|     liveData.observe(this) { it?.let { t -> action(t) } } | ||||
| } | ||||
| 
 | ||||
| /** NOTE: Only one observer at a time per value */ | ||||
| fun <T> LifecycleOwner.observeNullable(liveData: LiveData<T>, action: (t: T) -> Unit) { | ||||
|     liveData.removeObservers(this) | ||||
|     liveData.observe(this) { action(it) } | ||||
| } | ||||
| 
 | ||||
| sealed class Resource<out T> { | ||||
|     data class Success<out T>(val value: T) : Resource<T>() | ||||
|     data class Failure( | ||||
|         val isNetworkError: Boolean, | ||||
|         val errorCode: Int?, | ||||
|         val errorResponse: Any?, //ResponseBody | ||||
|         val errorString: String, | ||||
|     ) : Resource<Nothing>() | ||||
| 
 | ||||
|     data class Loading(val url: String? = null) : Resource<Nothing>() | ||||
| } | ||||
| 
 | ||||
| fun logError(throwable: Throwable) { | ||||
|     Log.d("ApiError", "-------------------------------------------------------------------") | ||||
|     Log.d("ApiError", "safeApiCall: " + throwable.localizedMessage) | ||||
|     Log.d("ApiError", "safeApiCall: " + throwable.message) | ||||
|     throwable.printStackTrace() | ||||
|     Log.d("ApiError", "-------------------------------------------------------------------") | ||||
| } | ||||
| 
 | ||||
| fun <T> normalSafeApiCall(apiCall: () -> T): T? { | ||||
|     return try { | ||||
|         apiCall.invoke() | ||||
|     } catch (throwable: Throwable) { | ||||
|         logError(throwable) | ||||
|         return null | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| suspend fun <T> suspendSafeApiCall(apiCall: suspend () -> T): T? { | ||||
|     return try { | ||||
|         apiCall.invoke() | ||||
|     } catch (throwable: Throwable) { | ||||
|         logError(throwable) | ||||
|         return null | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| fun Throwable.getAllMessages(): String { | ||||
|     return (this.localizedMessage ?: "") + (this.cause?.getAllMessages()?.let { "\n$it" } ?: "") | ||||
| } | ||||
| 
 | ||||
| fun Throwable.getStackTracePretty(showMessage: Boolean = true): String { | ||||
|     val prefix = if (showMessage) this.localizedMessage?.let { "\n$it" } ?: "" else "" | ||||
|     return prefix + this.stackTrace.joinToString( | ||||
|         separator = "\n" | ||||
|     ) { | ||||
|         "${it.fileName} ${it.lineNumber}" | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| fun <T> safeFail(throwable: Throwable): Resource<T> { | ||||
|     val stackTraceMsg = throwable.getStackTracePretty() | ||||
|     return Resource.Failure(false, null, null, stackTraceMsg) | ||||
| } | ||||
| 
 | ||||
| fun CoroutineScope.launchSafe( | ||||
|     context: CoroutineContext = EmptyCoroutineContext, | ||||
|     start: CoroutineStart = CoroutineStart.DEFAULT, | ||||
|     block: suspend CoroutineScope.() -> Unit | ||||
| ): Job { | ||||
|     val obj: suspend CoroutineScope.() -> Unit = { | ||||
|         try { | ||||
|             block() | ||||
|         } catch (throwable: Throwable) { | ||||
|             logError(throwable) | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     return this.launch(context, start, obj) | ||||
| } | ||||
| 
 | ||||
| fun<T> throwAbleToResource( | ||||
|     throwable: Throwable | ||||
| ): Resource<T> { | ||||
|     return when (throwable) { | ||||
|         is NullPointerException -> { | ||||
|             for (line in throwable.stackTrace) { | ||||
|                 if (line?.fileName?.endsWith("provider.kt", ignoreCase = true) == true) { | ||||
|                     return Resource.Failure( | ||||
|                         false, | ||||
|                         null, | ||||
|                         null, | ||||
|                         "NullPointerException at ${line.fileName} ${line.lineNumber}\nSite might have updated or added Cloudflare/DDOS protection" | ||||
|                     ) | ||||
|                 } | ||||
|             } | ||||
|             safeFail(throwable) | ||||
|         } | ||||
|         is SocketTimeoutException, is InterruptedIOException -> { | ||||
|             Resource.Failure( | ||||
|                 true, | ||||
|                 null, | ||||
|                 null, | ||||
|                 "Connection Timeout\nPlease try again later." | ||||
|             ) | ||||
|         } | ||||
|         is HttpException -> { | ||||
|             Resource.Failure( | ||||
|                 false, | ||||
|                 throwable.statusCode, | ||||
|                 null, | ||||
|                 throwable.message ?: "HttpException" | ||||
|             ) | ||||
|         } | ||||
|         is UnknownHostException -> { | ||||
|             Resource.Failure(true, null, null, "Cannot connect to server, try again later.\n${throwable.message}") | ||||
|         } | ||||
|         is ErrorLoadingException -> { | ||||
|             Resource.Failure( | ||||
|                 true, | ||||
|                 null, | ||||
|                 null, | ||||
|                 throwable.message ?: "Error loading, try again later." | ||||
|             ) | ||||
|         } | ||||
|         is NotImplementedError -> { | ||||
|             Resource.Failure(false, null, null, "This operation is not implemented.") | ||||
|         } | ||||
|         is SSLHandshakeException -> { | ||||
|             Resource.Failure( | ||||
|                 true, | ||||
|                 null, | ||||
|                 null, | ||||
|                 (throwable.message ?: "SSLHandshakeException") + "\nTry a VPN or DNS." | ||||
|             ) | ||||
|         } | ||||
|         is CancellationException -> { | ||||
|             throwable.cause?.let { | ||||
|                 throwAbleToResource(it) | ||||
|             } ?: safeFail(throwable) | ||||
|         } | ||||
|         else -> safeFail(throwable) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| suspend fun <T> safeApiCall( | ||||
|     apiCall: suspend () -> T, | ||||
| ): Resource<T> { | ||||
|     return withContext(Dispatchers.IO) { | ||||
|         try { | ||||
|             Resource.Success(apiCall.invoke()) | ||||
|         } catch (throwable: Throwable) { | ||||
|             logError(throwable) | ||||
|             throwAbleToResource(throwable) | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | @ -0,0 +1,16 @@ | |||
| package com.lagradost.cloudstream3.mvvm | ||||
| 
 | ||||
| import androidx.lifecycle.LifecycleOwner | ||||
| import androidx.lifecycle.LiveData | ||||
| 
 | ||||
| /** NOTE: Only one observer at a time per value */ | ||||
| fun <T> LifecycleOwner.observe(liveData: LiveData<T>, action: (t: T) -> Unit) { | ||||
|     liveData.removeObservers(this) | ||||
|     liveData.observe(this) { it?.let { t -> action(t) } } | ||||
| } | ||||
| 
 | ||||
| /** NOTE: Only one observer at a time per value */ | ||||
| fun <T> LifecycleOwner.observeNullable(liveData: LiveData<T>, action: (t: T) -> Unit) { | ||||
|     liveData.removeObservers(this) | ||||
|     liveData.observe(this) { action(it) } | ||||
| } | ||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue