From dc6af1df43cf3bed57585afdd7c05f2e2c314d8d Mon Sep 17 00:00:00 2001 From: Blatzar <46196380+Blatzar@users.noreply.github.com> Date: Sun, 11 Sep 2022 01:06:37 +0200 Subject: [PATCH 01/17] Fix GMPlayer audio --- .../cloudstream3/extractors/GMPlayer.kt | 16 +++++++++++----- 1 file changed, 11 insertions(+), 5 deletions(-) diff --git a/app/src/main/java/com/lagradost/cloudstream3/extractors/GMPlayer.kt b/app/src/main/java/com/lagradost/cloudstream3/extractors/GMPlayer.kt index e36a03d3..9078f389 100644 --- a/app/src/main/java/com/lagradost/cloudstream3/extractors/GMPlayer.kt +++ b/app/src/main/java/com/lagradost/cloudstream3/extractors/GMPlayer.kt @@ -4,6 +4,7 @@ import com.lagradost.cloudstream3.app import com.lagradost.cloudstream3.utils.ExtractorApi import com.lagradost.cloudstream3.utils.ExtractorLink import com.lagradost.cloudstream3.utils.M3u8Helper +import com.lagradost.cloudstream3.utils.Qualities class GMPlayer : ExtractorApi() { override val name = "GM Player" @@ -25,11 +26,16 @@ class GMPlayer : ExtractorApi() { data = mapOf("hash" to id, "r" to ref) ).parsed().videoSource ?: return null - return M3u8Helper.generateM3u8( - name, - m3u8, - ref, - headers = mapOf("accept" to "*/*") + return listOf( + ExtractorLink( + this.name, + this.name, + m3u8, + "", + Qualities.Unknown.value, + headers = mapOf("accept" to "*/*"), + isM3u8 = true + ) ) } From 9c5f2ef2f1de376bd170d59f049558cebe363eb9 Mon Sep 17 00:00:00 2001 From: Blatzar <46196380+Blatzar@users.noreply.github.com> Date: Sun, 11 Sep 2022 13:37:44 +0200 Subject: [PATCH 02/17] Fix GMPlayer --- .../main/java/com/lagradost/cloudstream3/extractors/GMPlayer.kt | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/app/src/main/java/com/lagradost/cloudstream3/extractors/GMPlayer.kt b/app/src/main/java/com/lagradost/cloudstream3/extractors/GMPlayer.kt index 9078f389..af02ee8a 100644 --- a/app/src/main/java/com/lagradost/cloudstream3/extractors/GMPlayer.kt +++ b/app/src/main/java/com/lagradost/cloudstream3/extractors/GMPlayer.kt @@ -31,7 +31,7 @@ class GMPlayer : ExtractorApi() { this.name, this.name, m3u8, - "", + ref, Qualities.Unknown.value, headers = mapOf("accept" to "*/*"), isM3u8 = true From 6f60298fc940ccc3dfb98ecc355e30fc4ef3f540 Mon Sep 17 00:00:00 2001 From: Blatzar <46196380+Blatzar@users.noreply.github.com> Date: Sun, 11 Sep 2022 13:43:05 +0200 Subject: [PATCH 03/17] Fix selecting homepage --- .../com/lagradost/cloudstream3/ui/home/HomeViewModel.kt | 9 +++++---- 1 file changed, 5 insertions(+), 4 deletions(-) diff --git a/app/src/main/java/com/lagradost/cloudstream3/ui/home/HomeViewModel.kt b/app/src/main/java/com/lagradost/cloudstream3/ui/home/HomeViewModel.kt index 1d6ed584..6254cb9b 100644 --- a/app/src/main/java/com/lagradost/cloudstream3/ui/home/HomeViewModel.kt +++ b/app/src/main/java/com/lagradost/cloudstream3/ui/home/HomeViewModel.kt @@ -272,12 +272,13 @@ class HomeViewModel : ViewModel() { if (!forceReload && api?.let { expandable[it.name]?.list?.list?.isNotEmpty() } == true) { return@launchSafe } - // If the plugin isn't loaded yet. (Does not set the key) - if (api == null) { - loadAndCancel(noneApi) - } else if (preferredApiName == noneApi.name) { + + if (preferredApiName == noneApi.name) { setKey(USER_SELECTED_HOMEPAGE_API, noneApi.name) loadAndCancel(noneApi) + // If the plugin isn't loaded yet. (Does not set the key) + } else if (api == null) { + loadAndCancel(noneApi) } else if (preferredApiName == randomApi.name) { val validAPIs = context?.filterProviderByPreferredMedia() if (validAPIs.isNullOrEmpty()) { From e830595c5f12e0212ed2c7d985c6e6fa14c0499c Mon Sep 17 00:00:00 2001 From: Blatzar <46196380+Blatzar@users.noreply.github.com> Date: Mon, 12 Sep 2022 16:00:27 +0200 Subject: [PATCH 04/17] Part 2 update for named seasons --- .../com/lagradost/cloudstream3/MainAPI.kt | 5 ++ .../ui/result/ResultViewModel2.kt | 57 +++++++++++-------- 2 files changed, 39 insertions(+), 23 deletions(-) diff --git a/app/src/main/java/com/lagradost/cloudstream3/MainAPI.kt b/app/src/main/java/com/lagradost/cloudstream3/MainAPI.kt index dc6cc454..80e22d8b 100644 --- a/app/src/main/java/com/lagradost/cloudstream3/MainAPI.kt +++ b/app/src/main/java/com/lagradost/cloudstream3/MainAPI.kt @@ -1118,6 +1118,11 @@ data class NextAiring( val unixTime: Long, ) +/** + * @param season To be mapped with episode season, not shown in UI if displaySeason is defined + * @param name To be shown next to the season like "Season $displaySeason $name" but if displaySeason is null then "$name" + * @param displaySeason What to be displayed next to the season name, if null then the name is the only thing shown. + * */ data class SeasonData( val season: Int, val name: String? = null, diff --git a/app/src/main/java/com/lagradost/cloudstream3/ui/result/ResultViewModel2.kt b/app/src/main/java/com/lagradost/cloudstream3/ui/result/ResultViewModel2.kt index 02fe60ca..b94143f1 100644 --- a/app/src/main/java/com/lagradost/cloudstream3/ui/result/ResultViewModel2.kt +++ b/app/src/main/java/com/lagradost/cloudstream3/ui/result/ResultViewModel2.kt @@ -1483,15 +1483,20 @@ class ResultViewModel2 : ViewModel() { 0 -> txt(R.string.no_season) else -> { val seasonNames = (currentResponse as? EpisodeResponse)?.seasonNames - val seasonData = - seasonNames.getSeason(indexer.season) - val suffix = seasonData?.name?.let { " $it" } ?: "" - txt( - R.string.season_format, - txt(R.string.season), - seasonData?.displaySeason ?: indexer.season, - suffix - ) + val seasonData = seasonNames.getSeason(indexer.season) + + // If displaySeason is null then only show the name! + if (seasonData?.name != null && seasonData.displaySeason == null) { + txt(seasonData.name) + } else { + val suffix = seasonData?.name?.let { " $it" } ?: "" + txt( + R.string.season_format, + txt(R.string.season), + seasonData?.displaySeason ?: indexer.season, + suffix + ) + } } } ) @@ -1598,7 +1603,7 @@ class ResultViewModel2 : ViewModel() { i.posterUrl, episode, null, - seasonData?.displaySeason ?: i.season, + if (seasonData != null) seasonData.displaySeason else i.season, i.data, loadResponse.apiName, id, @@ -1633,7 +1638,7 @@ class ResultViewModel2 : ViewModel() { if (!existingEpisodes.contains(id)) { existingEpisodes.add(id) val seasonIndex = episode.season?.minus(1) - val currentSeason = + val seasonData = loadResponse.seasonNames.getSeason(episode.season) val ep = @@ -1643,7 +1648,7 @@ class ResultViewModel2 : ViewModel() { episode.posterUrl, episodeIndex, seasonIndex, - currentSeason?.displaySeason ?: episode.season, + if (seasonData != null) seasonData.displaySeason else episode.season, episode.data, loadResponse.apiName, id, @@ -1747,16 +1752,17 @@ class ResultViewModel2 : ViewModel() { val seasonData = loadResponse.seasonNames.getSeason(seasonNumber) val fixedSeasonNumber = seasonData?.displaySeason ?: seasonNumber val suffix = seasonData?.name?.let { " $it" } ?: "" - - val name = - /*loadResponse.seasonNames?.firstOrNull { it.season == seasonNumber }?.name?.let { seasonData -> - txt(seasonData) - } ?:*/txt( - R.string.season_format, - txt(R.string.season), - fixedSeasonNumber, - suffix - ) + // If displaySeason is null then only show the name! + val name = if (seasonData?.name != null && seasonData.displaySeason == null) { + txt(seasonData.name) + } else { + txt( + R.string.season_format, + txt(R.string.season), + fixedSeasonNumber, + suffix + ) + } name to seasonNumber }) } @@ -1812,7 +1818,12 @@ class ResultViewModel2 : ViewModel() { } private fun loadTrailers(loadResponse: LoadResponse) = ioSafe { - _trailers.postValue(getTrailers(loadResponse, 3)) // we dont want to fetch too many trailers + _trailers.postValue( + getTrailers( + loadResponse, + 3 + ) + ) // we dont want to fetch too many trailers } private suspend fun getTrailers( From 7f475ba05992e39070df2894086ffe56bb8dc2a9 Mon Sep 17 00:00:00 2001 From: Blatzar <46196380+Blatzar@users.noreply.github.com> Date: Mon, 12 Sep 2022 17:22:48 +0200 Subject: [PATCH 05/17] Part 3 update for named seasons --- .../java/com/lagradost/cloudstream3/MainAPI.kt | 5 ++++- .../cloudstream3/ui/result/ResultViewModel2.kt | 17 ++++++++--------- 2 files changed, 12 insertions(+), 10 deletions(-) diff --git a/app/src/main/java/com/lagradost/cloudstream3/MainAPI.kt b/app/src/main/java/com/lagradost/cloudstream3/MainAPI.kt index 80e22d8b..d282f6dd 100644 --- a/app/src/main/java/com/lagradost/cloudstream3/MainAPI.kt +++ b/app/src/main/java/com/lagradost/cloudstream3/MainAPI.kt @@ -1203,9 +1203,12 @@ data class AnimeLoadResponse( override var backgroundPosterUrl: String? = null, ) : LoadResponse, EpisodeResponse +/** + * If episodes already exist appends the list. + * */ fun AnimeLoadResponse.addEpisodes(status: DubStatus, episodes: List?) { if (episodes.isNullOrEmpty()) return - this.episodes[status] = episodes + this.episodes[status] = (this.episodes[status] ?: emptyList()) + episodes } suspend fun MainAPI.newAnimeLoadResponse( diff --git a/app/src/main/java/com/lagradost/cloudstream3/ui/result/ResultViewModel2.kt b/app/src/main/java/com/lagradost/cloudstream3/ui/result/ResultViewModel2.kt index b94143f1..48919308 100644 --- a/app/src/main/java/com/lagradost/cloudstream3/ui/result/ResultViewModel2.kt +++ b/app/src/main/java/com/lagradost/cloudstream3/ui/result/ResultViewModel2.kt @@ -1592,8 +1592,8 @@ class ResultViewModel2 : ViewModel() { val idIndex = ep.key.id for ((index, i) in ep.value.withIndex()) { val episode = i.episode ?: (index + 1) - val id = mainId + episode + idIndex * 1000000 - if (!existingEpisodes.contains(episode)) { + val id = mainId + episode + idIndex * 1_000_000 + (i.season?.times(10_000) ?: 0) + if (!existingEpisodes.contains(id)) { existingEpisodes.add(id) val seasonData = loadResponse.seasonNames.getSeason(i.season) val eps = @@ -1602,7 +1602,7 @@ class ResultViewModel2 : ViewModel() { filterName(i.name), i.posterUrl, episode, - null, + seasonData?.season ?: i.season, if (seasonData != null) seasonData.displaySeason else i.season, i.data, loadResponse.apiName, @@ -1615,7 +1615,7 @@ class ResultViewModel2 : ViewModel() { mainId ) - val season = eps.season ?: 0 + val season = eps.seasonIndex ?: 0 val indexer = EpisodeIndexer(ep.key, season) episodes[indexer]?.add(eps) ?: run { episodes[indexer] = mutableListOf(eps) @@ -1630,14 +1630,13 @@ class ResultViewModel2 : ViewModel() { mutableMapOf() val existingEpisodes = HashSet() for ((index, episode) in loadResponse.episodes.sortedBy { - (it.season?.times(10000) ?: 0) + (it.episode ?: 0) + (it.season?.times(10_000) ?: 0) + (it.episode ?: 0) }.withIndex()) { val episodeIndex = episode.episode ?: (index + 1) val id = - mainId + (episode.season?.times(100000) ?: 0) + episodeIndex + 1 + mainId + (episode.season?.times(100_000) ?: 0) + episodeIndex + 1 if (!existingEpisodes.contains(id)) { existingEpisodes.add(id) - val seasonIndex = episode.season?.minus(1) val seasonData = loadResponse.seasonNames.getSeason(episode.season) @@ -1647,7 +1646,7 @@ class ResultViewModel2 : ViewModel() { filterName(episode.name), episode.posterUrl, episodeIndex, - seasonIndex, + seasonData?.season ?: episode.season, if (seasonData != null) seasonData.displaySeason else episode.season, episode.data, loadResponse.apiName, @@ -1660,7 +1659,7 @@ class ResultViewModel2 : ViewModel() { mainId ) - val season = episode.season ?: 0 + val season = ep.seasonIndex ?: 0 val indexer = EpisodeIndexer(DubStatus.None, season) episodes[indexer]?.add(ep) ?: kotlin.run { From d6aa0e576c98656084cccb194868ad16cea85265 Mon Sep 17 00:00:00 2001 From: Blatzar <46196380+Blatzar@users.noreply.github.com> Date: Mon, 12 Sep 2022 17:58:03 +0200 Subject: [PATCH 06/17] Fix create account dialog on RTL locale & remove update cache --- .../cloudstream3/utils/InAppUpdater.kt | 13 +- app/src/main/res/layout/add_account_input.xml | 227 +++++++++--------- 2 files changed, 125 insertions(+), 115 deletions(-) diff --git a/app/src/main/java/com/lagradost/cloudstream3/utils/InAppUpdater.kt b/app/src/main/java/com/lagradost/cloudstream3/utils/InAppUpdater.kt index 42d200d0..54492e6d 100644 --- a/app/src/main/java/com/lagradost/cloudstream3/utils/InAppUpdater.kt +++ b/app/src/main/java/com/lagradost/cloudstream3/utils/InAppUpdater.kt @@ -200,16 +200,23 @@ class InAppUpdater { private suspend fun Activity.downloadUpdate(url: String): Boolean { try { Log.d(LOG_TAG, "Downloading update: $url") + val appUpdateName = "CloudStream" + val appUpdateSuffix = "apk" - val localContext = this + // Delete all old updates + this.cacheDir.listFiles()?.filter { + it.name.startsWith(appUpdateName) && it.extension == appUpdateSuffix + }?.forEach { + it.deleteOnExit() + } - val downloadedFile = File.createTempFile("CloudStream", ".apk") + val downloadedFile = File.createTempFile(appUpdateName, ".$appUpdateSuffix") val sink: BufferedSink = downloadedFile.sink().buffer() updateLock.withLock { sink.writeAll(app.get(url).body.source()) sink.close() - openApk(localContext, Uri.fromFile(downloadedFile)) + openApk(this, Uri.fromFile(downloadedFile)) } return true } catch (e: Exception) { diff --git a/app/src/main/res/layout/add_account_input.xml b/app/src/main/res/layout/add_account_input.xml index 1471af9c..ea48a80f 100644 --- a/app/src/main/res/layout/add_account_input.xml +++ b/app/src/main/res/layout/add_account_input.xml @@ -1,131 +1,134 @@ - - - - - - - + xmlns:app="http://schemas.android.com/apk/res-auto" + xmlns:tools="http://schemas.android.com/tools" + android:layout_width="match_parent" + android:layout_height="match_parent" + android:orientation="vertical"> + android:layout_width="match_parent" + android:layout_height="wrap_content" + android:layout_marginTop="20dp" + android:layout_marginBottom="10dp" + android:orientation="horizontal"> - + + android:paddingStart="?android:attr/listPreferredItemPaddingStart" + android:paddingEnd="?android:attr/listPreferredItemPaddingEnd" + android:textColor="?attr/textColor" + android:textSize="20sp" + android:textStyle="bold" + tools:text="Test" /> - - - + + + + android:layout_height="wrap_content" + android:autofillHints="username" + android:hint="@string/example_username" + android:inputType="text" + android:nextFocusLeft="@id/apply_btt" + android:nextFocusRight="@id/cancel_btt" + android:nextFocusDown="@id/login_email_input" + android:requiresFadingEdge="vertical" + android:textColorHint="?attr/grayTextColor" + tools:ignore="LabelFor" /> + + + + + + + + + + + android:id="@+id/apply_btt" + style="@style/WhiteButton" + android:layout_width="wrap_content" + android:layout_gravity="center_vertical|end" + android:text="@string/login" /> + android:id="@+id/cancel_btt" + style="@style/BlackButton" + android:layout_width="wrap_content" + android:layout_gravity="center_vertical|end" + android:text="@string/sort_cancel" /> \ No newline at end of file From beca2b7b8479aebd13973a5986571e1ccd4d9d5e Mon Sep 17 00:00:00 2001 From: Hexated <37908684+hexated@users.noreply.github.com> Date: Tue, 13 Sep 2022 02:38:33 +0700 Subject: [PATCH 07/17] [extractor] added Gdriveplayer (#99) * added Gdriveplayer * added Sbflix * added DoodWfExtractor * fixed StreamSB --- .../cloudstream3/extractors/DoodExtractor.kt | 4 + .../cloudstream3/extractors/Gdriveplayer.kt | 178 ++++++++++++++++++ .../cloudstream3/extractors/StreamSB.kt | 7 +- .../cloudstream3/utils/ExtractorApi.kt | 13 ++ 4 files changed, 201 insertions(+), 1 deletion(-) create mode 100644 app/src/main/java/com/lagradost/cloudstream3/extractors/Gdriveplayer.kt diff --git a/app/src/main/java/com/lagradost/cloudstream3/extractors/DoodExtractor.kt b/app/src/main/java/com/lagradost/cloudstream3/extractors/DoodExtractor.kt index c5eaf40e..7ec1fb22 100644 --- a/app/src/main/java/com/lagradost/cloudstream3/extractors/DoodExtractor.kt +++ b/app/src/main/java/com/lagradost/cloudstream3/extractors/DoodExtractor.kt @@ -7,6 +7,10 @@ import com.lagradost.cloudstream3.utils.Qualities import com.lagradost.cloudstream3.utils.getQualityFromName import kotlinx.coroutines.delay +class DoodWfExtractor : DoodLaExtractor() { + override var mainUrl = "https://dood.wf" +} + class DoodCxExtractor : DoodLaExtractor() { override var mainUrl = "https://dood.cx" } diff --git a/app/src/main/java/com/lagradost/cloudstream3/extractors/Gdriveplayer.kt b/app/src/main/java/com/lagradost/cloudstream3/extractors/Gdriveplayer.kt new file mode 100644 index 00000000..d2e56bf1 --- /dev/null +++ b/app/src/main/java/com/lagradost/cloudstream3/extractors/Gdriveplayer.kt @@ -0,0 +1,178 @@ +package com.lagradost.cloudstream3.extractors + +import com.fasterxml.jackson.annotation.JsonProperty +import com.lagradost.cloudstream3.* +import com.lagradost.cloudstream3.utils.* +import org.jsoup.nodes.Element +import java.security.DigestException +import java.security.MessageDigest +import javax.crypto.Cipher +import javax.crypto.spec.IvParameterSpec +import javax.crypto.spec.SecretKeySpec + +class Gdriveplayerapi: Gdriveplayer() { + override val mainUrl: String = "https://gdriveplayerapi.com" +} + +class Gdriveplayerapp: Gdriveplayer() { + override val mainUrl: String = "https://gdriveplayer.app" +} + +class Gdriveplayerfun: Gdriveplayer() { + override val mainUrl: String = "https://gdriveplayer.fun" +} + +class Gdriveplayerio: Gdriveplayer() { + override val mainUrl: String = "https://gdriveplayer.io" +} + +class Gdriveplayerme: Gdriveplayer() { + override val mainUrl: String = "https://gdriveplayer.me" +} + +class Gdriveplayerbiz: Gdriveplayer() { + override val mainUrl: String = "https://gdriveplayer.biz" +} + +class Gdriveplayerorg: Gdriveplayer() { + override val mainUrl: String = "https://gdriveplayer.org" +} + +class Gdriveplayerus: Gdriveplayer() { + override val mainUrl: String = "https://gdriveplayer.us" +} + +class Gdriveplayerco: Gdriveplayer() { + override val mainUrl: String = "https://gdriveplayer.co" +} + +open class Gdriveplayer : ExtractorApi() { + override val name = "Gdrive" + override val mainUrl = "https://gdriveplayer.to" + override val requiresReferer = false + + private fun unpackJs(script: Element): String? { + return script.select("script").find { it.data().contains("eval(function(p,a,c,k,e,d)") } + ?.data()?.let { getAndUnpack(it) } + } + + private fun String.decodeHex(): ByteArray { + check(length % 2 == 0) { "Must have an even length" } + return chunked(2) + .map { it.toInt(16).toByte() } + .toByteArray() + } + + // https://stackoverflow.com/a/41434590/8166854 + private fun GenerateKeyAndIv( + password: ByteArray, + salt: ByteArray, + hashAlgorithm: String = "MD5", + keyLength: Int = 32, + ivLength: Int = 16, + iterations: Int = 1 + ): List? { + + val md = MessageDigest.getInstance(hashAlgorithm) + val digestLength = md.digestLength + val targetKeySize = keyLength + ivLength + val requiredLength = (targetKeySize + digestLength - 1) / digestLength * digestLength + val generatedData = ByteArray(requiredLength) + var generatedLength = 0 + + try { + md.reset() + + while (generatedLength < targetKeySize) { + if (generatedLength > 0) + md.update( + generatedData, + generatedLength - digestLength, + digestLength + ) + + md.update(password) + md.update(salt, 0, 8) + md.digest(generatedData, generatedLength, digestLength) + + for (i in 1 until iterations) { + md.update(generatedData, generatedLength, digestLength) + md.digest(generatedData, generatedLength, digestLength) + } + + generatedLength += digestLength + } + return listOf( + generatedData.copyOfRange(0, keyLength), + generatedData.copyOfRange(keyLength, targetKeySize) + ) + } catch (e: DigestException) { + return null + } + } + + private fun cryptoAESHandler( + data: AesData, + pass: ByteArray, + encrypt: Boolean = true + ): String? { + val (key, iv) = GenerateKeyAndIv(pass, data.s.decodeHex()) ?: return null + val cipher = Cipher.getInstance("AES/CBC/NoPadding") + return if (!encrypt) { + cipher.init(Cipher.DECRYPT_MODE, SecretKeySpec(key, "AES"), IvParameterSpec(iv)) + String(cipher.doFinal(base64DecodeArray(data.ct))) + } else { + cipher.init(Cipher.DECRYPT_MODE, SecretKeySpec(key, "AES"), IvParameterSpec(iv)) + base64Encode(cipher.doFinal(data.ct.toByteArray())) + + } + } + + private fun Regex.first(str: String): String? { + return find(str)?.groupValues?.getOrNull(1) + } + + override suspend fun getUrl( + url: String, + referer: String?, + subtitleCallback: (SubtitleFile) -> Unit, + callback: (ExtractorLink) -> Unit + ) { + val document = app.get(url).document + + val eval = unpackJs(document)?.replace("\\", "") ?: return + val data = AppUtils.tryParseJson(Regex("data='(\\S+?)'").first(eval)) ?: return + val password = Regex("null,['|\"](\\w+)['|\"]").first(eval) + ?.split(Regex("\\D+")) + ?.joinToString("") { + Char(it.toInt()).toString() + }.let { Regex("var pass = \"(\\S+?)\"").first(it ?: return)?.toByteArray() } + ?: throw ErrorLoadingException("can't find password") + val decryptedData = + cryptoAESHandler(data, password, false)?.let { getAndUnpack(it) }?.replace("\\", "") + ?.substringAfter("sources:[")?.substringBefore("],") + + Regex("\"file\":\"(\\S+?)\".*?res=(\\d+)").findAll(decryptedData ?: return).map { + it.groupValues[1] to it.groupValues[2] + }.toList().distinctBy { it.second }.map { (link, quality) -> + callback.invoke( + ExtractorLink( + source = this.name, + name = this.name, + url = "${httpsify(link)}&res=$quality", + referer = mainUrl, + quality = quality.toIntOrNull() ?: Qualities.Unknown.value, + headers = mapOf("Range" to "bytes=0-") + ) + ) + } + + } + + data class AesData( + @JsonProperty("ct") val ct: String, + @JsonProperty("iv") val iv: String, + @JsonProperty("s") val s: String + ) + +} \ No newline at end of file diff --git a/app/src/main/java/com/lagradost/cloudstream3/extractors/StreamSB.kt b/app/src/main/java/com/lagradost/cloudstream3/extractors/StreamSB.kt index a933c484..461f56d0 100644 --- a/app/src/main/java/com/lagradost/cloudstream3/extractors/StreamSB.kt +++ b/app/src/main/java/com/lagradost/cloudstream3/extractors/StreamSB.kt @@ -7,6 +7,11 @@ import com.lagradost.cloudstream3.utils.ExtractorApi import com.lagradost.cloudstream3.utils.ExtractorLink import com.lagradost.cloudstream3.utils.M3u8Helper +class Sbflix : StreamSB() { + override var mainUrl = "https://sbflix.xyz" + override var name = "Sbflix" +} + class Vidgomunime : StreamSB() { override var mainUrl = "https://vidgomunime.xyz" } @@ -111,7 +116,7 @@ open class StreamSB : ExtractorApi() { }.first() val bytes = id.toByteArray() val bytesToHex = bytesToHex(bytes) - val master = "$mainUrl/sources43/6d6144797752744a454267617c7c${bytesToHex.lowercase()}7c7c4e61755a56456f34385243727c7c73747265616d7362/6b4a33767968506e4e71374f7c7c343837323439333133333462353935333633373836643638376337633462333634663539343137373761333635313533333835333763376333393636363133393635366136323733343435323332376137633763373337343732363536313664373336327c7c504d754478413835306633797c7c73747265616d7362" + val master = "$mainUrl/sources44/6d6144797752744a454267617c7c${bytesToHex.lowercase()}7c7c4e61755a56456f34385243727c7c73747265616d7362/6b4a33767968506e4e71374f7c7c343837323439333133333462353935333633373836643638376337633462333634663539343137373761333635313533333835333763376333393636363133393635366136323733343435323332376137633763373337343732363536313664373336327c7c504d754478413835306633797c7c73747265616d7362" val headers = mapOf( "watchsb" to "streamsb", ) diff --git a/app/src/main/java/com/lagradost/cloudstream3/utils/ExtractorApi.kt b/app/src/main/java/com/lagradost/cloudstream3/utils/ExtractorApi.kt index 7a65df30..ae3d8c06 100644 --- a/app/src/main/java/com/lagradost/cloudstream3/utils/ExtractorApi.kt +++ b/app/src/main/java/com/lagradost/cloudstream3/utils/ExtractorApi.kt @@ -236,6 +236,7 @@ val extractorApis: MutableList = arrayListOf( Ssbstream(), Sbthe(), Vidgomunime(), + Sbflix(), Fastream(), @@ -269,6 +270,7 @@ val extractorApis: MutableList = arrayListOf( DoodWsExtractor(), DoodShExtractor(), DoodWatchExtractor(), + DoodWfExtractor(), AsianLoad(), @@ -321,6 +323,17 @@ val extractorApis: MutableList = arrayListOf( Mvidoo(), Streamplay(), + Gdriveplayerapi(), + Gdriveplayerapp(), + Gdriveplayerfun(), + Gdriveplayerio(), + Gdriveplayerme(), + Gdriveplayerbiz(), + Gdriveplayerorg(), + Gdriveplayerus(), + Gdriveplayerco(), + Gdriveplayer(), + YoutubeExtractor(), YoutubeShortLinkExtractor(), YoutubeMobileExtractor(), From 2bd4d67c5d70801bf250ab20591714ba480f8301 Mon Sep 17 00:00:00 2001 From: Blatzar <46196380+Blatzar@users.noreply.github.com> Date: Tue, 13 Sep 2022 10:52:51 +0200 Subject: [PATCH 08/17] Fixed random timeout crashes --- app/build.gradle | 2 +- .../java/com/lagradost/cloudstream3/mvvm/ArchComponentExt.kt | 5 +++-- .../com/lagradost/cloudstream3/network/RequestsHelper.kt | 3 --- 3 files changed, 4 insertions(+), 6 deletions(-) diff --git a/app/build.gradle b/app/build.gradle index ed641a0a..b80c820f 100644 --- a/app/build.gradle +++ b/app/build.gradle @@ -170,7 +170,7 @@ dependencies { // Networking // implementation "com.squareup.okhttp3:okhttp:4.9.2" // implementation "com.squareup.okhttp3:okhttp-dnsoverhttps:4.9.1" - implementation 'com.github.Blatzar:NiceHttp:0.3.2' + implementation 'com.github.Blatzar:NiceHttp:0.3.3' // Util to skip the URI file fuckery 🙏 implementation "com.github.tachiyomiorg:unifile:17bec43" diff --git a/app/src/main/java/com/lagradost/cloudstream3/mvvm/ArchComponentExt.kt b/app/src/main/java/com/lagradost/cloudstream3/mvvm/ArchComponentExt.kt index a64f0d8d..5c3276fa 100644 --- a/app/src/main/java/com/lagradost/cloudstream3/mvvm/ArchComponentExt.kt +++ b/app/src/main/java/com/lagradost/cloudstream3/mvvm/ArchComponentExt.kt @@ -7,6 +7,7 @@ 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 @@ -157,7 +158,7 @@ suspend fun safeApiCall( } safeFail(throwable) } - is SocketTimeoutException -> { + is SocketTimeoutException, is InterruptedIOException -> { Resource.Failure( true, null, @@ -192,7 +193,7 @@ suspend fun safeApiCall( true, null, null, - (throwable.message ?: "SSLHandshakeException") + "\nTry again later." + (throwable.message ?: "SSLHandshakeException") + "\nTry a VPN or DNS." ) } else -> safeFail(throwable) diff --git a/app/src/main/java/com/lagradost/cloudstream3/network/RequestsHelper.kt b/app/src/main/java/com/lagradost/cloudstream3/network/RequestsHelper.kt index 03ec6ae8..13299002 100644 --- a/app/src/main/java/com/lagradost/cloudstream3/network/RequestsHelper.kt +++ b/app/src/main/java/com/lagradost/cloudstream3/network/RequestsHelper.kt @@ -5,15 +5,12 @@ import androidx.preference.PreferenceManager import com.lagradost.cloudstream3.R import com.lagradost.cloudstream3.USER_AGENT import com.lagradost.nicehttp.Requests -import com.lagradost.nicehttp.getCookies import com.lagradost.nicehttp.ignoreAllSSLErrors import okhttp3.Cache import okhttp3.Headers import okhttp3.Headers.Companion.toHeaders import okhttp3.OkHttpClient -import okhttp3.Request import java.io.File -import java.util.concurrent.TimeUnit fun Requests.initClient(context: Context): OkHttpClient { From 4f8f5024cb9e588f41b6ef752777a783cdc8b8d0 Mon Sep 17 00:00:00 2001 From: Blatzar <46196380+Blatzar@users.noreply.github.com> Date: Tue, 13 Sep 2022 11:06:11 +0200 Subject: [PATCH 09/17] TV Navigation improvements --- app/src/main/res/layout/fragment_plugins.xml | 3 +++ app/src/main/res/layout/fragment_search.xml | 4 +++- app/src/main/res/layout/home_select_mainpage.xml | 6 ++++-- 3 files changed, 10 insertions(+), 3 deletions(-) diff --git a/app/src/main/res/layout/fragment_plugins.xml b/app/src/main/res/layout/fragment_plugins.xml index 54eae80f..15e0d2f9 100644 --- a/app/src/main/res/layout/fragment_plugins.xml +++ b/app/src/main/res/layout/fragment_plugins.xml @@ -102,18 +102,21 @@ style="@style/RoundedSelectableButton" android:nextFocusLeft="@id/home_select_cartoons" + android:nextFocusRight="@id/home_select_livestreams" android:text="@string/documentaries" /> - + Date: Tue, 13 Sep 2022 19:50:25 +0200 Subject: [PATCH 10/17] =?UTF-8?q?Made=20repository=20focusable=20on=20Fire?= =?UTF-8?q?Stick=20=F0=9F=92=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../cloudstream3/ui/settings/extensions/RepoAdapter.kt | 2 +- app/src/main/res/layout/repository_item_tv.xml | 2 ++ 2 files changed, 3 insertions(+), 1 deletion(-) diff --git a/app/src/main/java/com/lagradost/cloudstream3/ui/settings/extensions/RepoAdapter.kt b/app/src/main/java/com/lagradost/cloudstream3/ui/settings/extensions/RepoAdapter.kt index 2dc302ee..e90166a8 100644 --- a/app/src/main/java/com/lagradost/cloudstream3/ui/settings/extensions/RepoAdapter.kt +++ b/app/src/main/java/com/lagradost/cloudstream3/ui/settings/extensions/RepoAdapter.kt @@ -76,7 +76,7 @@ class RepoAdapter( imageClickCallback(repositoryData) } - itemView.setOnClickListener { + itemView.repository_item_root?.setOnClickListener { clickCallback(repositoryData) } itemView.main_text?.text = repositoryData.name diff --git a/app/src/main/res/layout/repository_item_tv.xml b/app/src/main/res/layout/repository_item_tv.xml index 881fce63..fbe18199 100644 --- a/app/src/main/res/layout/repository_item_tv.xml +++ b/app/src/main/res/layout/repository_item_tv.xml @@ -8,6 +8,8 @@ android:background="@drawable/outline_drawable" android:nextFocusRight="@id/action_button" android:orientation="horizontal" + android:clickable="true" + android:focusable="true" android:padding="12dp"> Date: Wed, 14 Sep 2022 11:43:30 +0200 Subject: [PATCH 11/17] arabicSubs100% [SANCTI] (#101) --- app/src/main/res/values-ar/strings.xml | 436 +++++++++++++++++++------ 1 file changed, 344 insertions(+), 92 deletions(-) diff --git a/app/src/main/res/values-ar/strings.xml b/app/src/main/res/values-ar/strings.xml index 2eaa9af5..23b9af9b 100644 --- a/app/src/main/res/values-ar/strings.xml +++ b/app/src/main/res/values-ar/strings.xml @@ -5,57 +5,65 @@ ملصق الحلقة الملصق الرئيسي التالي عشوائي + ارجع للخلف + تغيير المصدر معاينة الخلفية سرعة (%.2fx) Rated: %.1f + !تم إيجاد تحديث جديد\n%s -> %s %d دقيقة CloudStream + تشغيل بواسطة CloudStream الصفحة الرئيسية - بحث + البحث التحميلات الإعدادات - …بحث + …بحث + بحث %s… + لايوجد بيانات المزيد من الخيارات الحلقة التالية - النواع + أنواع شارك - فتح في الويب + فتح في الويب تخطي التحميل …تحميل - مشاهدة + أشاهده في الانتظار مكتمل - إسقاط - تخطط للمشاهدة + مهمل + أخطط لمشاهدته لا شيء إعادة المشاهدة - + مشاهدة الفيلم - تشغيل التورنت + تشغيل بث حي + تشغيل تورنت المصادر الترجمة …إعادة محاولة الاتصال ارجع للخلف تشغيل الحلقة - + تحميل تم التنزيل - جارى التحميل + جاري التنزيل توقف التنزيل مؤقتًا بدأ التنزيل - التحميل فشل + فشل التنزيل تم إلغاء التنزيل - تنزيل تم + تم التنزيل + تشغيل خطأ في تحميل الرابط التخزين الداخلي @@ -73,12 +81,18 @@ إخفاء تشغيل معلومات - تصفية المواقع المفضلة + تصفية الاشارات المرجعية إشارات مرجعية حذف + إعداد حالة المشاهدة تطبيق إلغاء - سرعة المشغل + نسخ + إغلاق + مسح + حفظ + + سرعة المُشغل إعدادات الترجمة لون الخط @@ -90,65 +104,97 @@ الخط حجم الخط - ابحث باستخدام المصادر + البحث باستخدام المصادر البحث باستخدام الأنواع - %d البنينيس المعطاة الى المطورين - لم يتم إعطاء بنين + %d الموزات المعطاة الى المطورين + لم يتم إعطاء موز تحديد اللغة تلقائيًا تحميل اللغات - اضغط بإستمرار لإعادة التعيين - استمر في المشاهدة + لغة الترجمة + إضغط بإستمرار لإعادة التعيين للإعدادات الافتراضية + إستيراد خطوط بوضعها هنا %s + متابعة المشاهدة حذف مزيد من المعلومات - قد تكون هناك حاجة إلى شبكة ظاهرية خاصة لكي يعمل هذا المزود بشكل صحيح + قد تكون هناك حاجة إلى VPN لكي يعمل هذا المزود بشكل صحيح هذا المزود هو تورنت ، يوصى باستخدام شبكة ظاهرية خاصة - + .سيظهر إعلان مدته خمسة عشر ثانية إذا لم يتم توفير الفيديو في الموقع - + الوصف لم يتم العثور على وصف لم يتم العثور على وصف + 🐈 logcat عرض + نافذة منبثقة - يستمر في التشغيل في مشغل مصغر فوق التطبيقات الأخرى - زر تغيير حجم المشغل - قم بإزالة الحدود السوداء + يستمر في التشغيل في مُشغل مصغر فوق التطبيقات الأخرى + زر تغيير حجم المُشغل + إزالة الحدود السوداء الترجمة - إعدادات ترجمة المشغل + إعدادات ترجمة المُشغل + ترجمة كروم كاست + إعدادات ترجمة كروم كاست + وضع إيغنغرافي - يضيف خيار السرعة في المشغل - اسحب للسعي - اسحب إلى اليسار أو اليمين للتحكم في الوقت في مشغل الفيديو - اسحب لتغيير الإعدادات - اسحب على الجانب الأيسر أو الأيمن لتغيير السطوع أو مستوى الصوت - انقر مرتين للسعي للأمام أو للخلف - اضغط مرتين لإيقاف مؤقت - اضغط مرتين على الجانب الأيمن أو الأيسر للسعي للأمام أو للخلف - اضغط في الوسط لإيقاف مؤقت + يضيف خيار السرعة في المُشغل + السحب للسعي + إسحب إلى اليسار أو اليمين للتحكم في الوقت في مُشغل الفيديو + السحب لتغيير الإعدادات + إسحب على الجانب الأيسر أو الأيمن لتغيير السطوع أو مستوى الصوت + + تشغيل الحلقة التالية تلقائيًا + تبدأ الحلقة التالية عندما تنتهي الحالية + + النقر مرتان للسعي للأمام أو للخلف + الضغط مرتان لإيقاف مؤقت + التحكم في سعي المُشغل + إضغط مرتين على الجانب الأيمن أو الأيسر للسعي للأمام أو للخلف + إضغط في الوسط لإيقاف مؤقت استخدم سطوع النظام - استخدم سطوع النظام في مشغل التطبيق بدلاً من التراكب الداكن + استخدم سطوع النظام في مُشغل التطبيق بدلاً من التراكب الداكن + + تحديث تقدم المشاهدة + مزامنة التقدم في الحلقة الحالية تلقائيًا + + إسترجاع البيانات من نسخة إحتياطية + + نسخ إحتياطي + تم تحميل ملف النسخة الاحتياطية + فشل استيراد البيانات من الملف %s + تم تخزين البيانات بنجاح + إذن الوصول الي ذاكرة التخزين مفقود, من فضلك حاول مجددا + فشل إنشاء نسخة احتياطية %s بحث + الحسابات + التحديثات والنسخ الاحتياطية + معلومات البحث المتقدم يعطيك نتائج البحث مفصولة عن طريق المزود - إرسال البيانات عن الأعطال فقط + إرسال البيانات عند الأعطال فقط لا ترسل أي بيانات - عرض حلقة فلر لأنيمي + عرض حلقات الفلر للأنمي + عرض المقاطع الدعائية + عرض ملصقات من kitsu + إخفاء جودة الفيديو المختارة من نتائج البحث + + تحديث الإضافات تلقائيًا التحديث التلقائي - ابحث تلقائيًا عن التحديثات الجديدة عند البداية + البحث تلقائيًا عن التحديثات الجديدة عند البداية التحديث إلى الاصدارات التجريبيه (بيتا) - ابحث عن التحديثات التجريبية بدلاً من الإصدارات الكاملة فقط + البحث عن التحديثات التجريبية بدلاً من الإصدارات الكاملة فقط Github - تطبيق رواية خفيف من نفس المطورين - تطبيق Anime من نفس المطورين - انضم إلى إلديسكورد - أعط موزة للمطورين - أعط الموز + تطبيق روايات خفيف من نفس المطورين + تطبيق أنمي من نفس المطورين + انضم إلى الديسكورد + إعط موزة للمطورين + الموز المُعطي لغة التطبيق @@ -158,23 +204,24 @@ تشغيل الحلقة إعادة التعيين إلى القيمة الافتراضية عذرا ، تعطل التطبيق. سيتم إرسال تقرير خطأ مجهول إلى المطورين - + + موسم لا موسم حلقة حلقات - S - E + ح + م لم يتم العثور على أي حلقات - حذف ملف + حذف الملف حذف إيقاف مؤقت أكمل - -30 - +30 - This will permanently delete %s\nAre you sure? - %dد\nمتبقية + -٣٠ + +٣٠ + سوف يتم الحذف نهائيا %s\nهل أنت متأكد? + %dm\nمتبقية جاري التنفيذ اكتمل @@ -189,28 +236,39 @@ الترجمة ليست موجودة الإفتراضي - حر + فارغ مستخدم - تطبيق + التطبيق + أفلام مسلسلات رسوم متحركة - انمي + أنمي تورنت - الافلام الوثائقية - OVA - + أفلام وثائقية + أوفا + دراما آسيوية + بث حي + +18 + أخري + فيلم - مسلسلات + مسلسل كارتون + @string/anime + @string/ova تورنت - وثائقي - + وثائقي + دراما آسيوية + بث حي + +18 + فيديو + خطأ في المصدر خطأ بعيد خطأ في جهاز العرض - خطأ غير متوقع في مشغل + خطأ غير متوقع في المُشغل خطأ في التنزيل ، تحقق من أذونات التخزين حلقة كروم كاست @@ -218,10 +276,21 @@ تشغيل في التطبيق VLC تشغيل في تشغيل في الويب - انسخ الرابط + نسخ الرابط التحميل التلقائي - تحميل المرآة + تحميل بجودات مختلفة إعادة تحميل الروابط + تحميل الترجمة + + ملصق الجودة + ملصق مدبلج + ملصق مترجم + العنوان + show_hd_key + show_dub_key + show_sub_key + show_title_key + التحكم في عناصر الواجهة علي الملصق لم يتم العثور على تحديث تحقق من التحديثات @@ -229,61 +298,244 @@ قفل تغيير الحجم مصدر - OP تخطي + تخطي المقدمة لا تظهر مرة أخرى + تخطي هذا التحديث تحديث جودة المشاهدة المفضلة + أقصي عدد حروف لعنوان مُشغل الفيديو + أبعاد مُشغل الفيديو + حجم ذاكرة التخزين المؤقت للفيديو + طول التخزين المؤقت + التخزين المؤقت للفيديو علي القرص + مسح التخزين المؤقت للصورة والفيديو + + سوف يسبب بعض الأعطال إذا تم إعداد قيمة عالية جدًا. لا\تغير القيمة إذا كان لديك ذاكرة تخزين عشوائية منخفضة كتلفاز أندرويد أو هاتف قديم + قد يسبب مشاكل مع حجم ذاكرة التخزين المنخفضة كأجهزة تلفاز الأندرويد إذا أعددت قيمة عالية جدًا + DNS فوق HTTPS - مفيد لتجاوز كتل مزود خدمة الإنترنت - - موقع التنزيل - + مفيد لتجاوز حجب مزود خدمة الإنترنت + + نسخ موقع + حذف موقع + إضافة نسخة من موقع موجود, بعنوان رابط مختلف + + مسار التنزيل + + عنوان رابط سيرفر Nginx + عرض أنمي مدبلج / مترجم - تناسب الشاشة + ملائمة الشاشة امتداد تكبير + إخلاء مسؤولية + legal_notice_key + Any legal issues regarding the content on this application + should be taken up with the actual file hosts and providers themselves as we are not affiliated with them. + + In case of copyright infringement, please directly contact the responsible parties or the streaming websites. + + The app is purely for educational and personal use. + + CloudStream 3 does not host any content on the app, and has no control over what media is put up or taken down. + CloudStream 3 functions like any other search engine, such as Google. CloudStream 3 does not host, upload or + manage any videos, films or content. It simply crawls, aggregates and displayes links in a convenient, + user-friendly interface. + + It merely scrapes 3rd-party websites that are publicly accessable via any regular web browser. It is the + responsibility of user to avoid any actions that might violate the laws governing his/her locality. Use + CloudStream 3 at your own risk. + عام + زر العشوائي + إظهار زر العشوائي علي الشاشة الرئيسية لغات الموفر - واجهة التطبيق - النوع المفضل من المشاهدة - - أوتوماتيك - واجهة خاصة بتلفاز - واجهة خاصة بهاتف - + واجهة التطبيق + المحتوي المفضل + تفعيل محتوي البالغين داخل المزودين المدعومين + فك تشفير الترجمة + المصادر + الواجهة + + + أوتوماتيك + واجهة تلفاز + واجهة هاتف + واجهة محاكي + اللون الأساسي مظهر التطبيق - - الحساب + موضع عنوان الملصق + وضع العنوان تحت الملصق + + + + anilist_key + mal_key + opensubtitles_key + nginx_key + password123 + MyCoolUsername + hello@world.com + 127.0.0.1 + MyCoolSite + example.com + Language code (en) + + + %s %s + حساب تسجيل خروج تسجيل الدخول تبديل الحساب إضافة حساب - Add tracking + إنشاء حساب + إضافة تتبع + تم إضافة %s + مزامنة + مقيّم + %d / 10 + /?? + /%d + تم توثيق %s + فشل توثيق %s + لا شيء عادي الكل - ماكس + الحد الاقصي الحد الأدنى - الخطوط العريضة - النمط المكتئب + @string/none + الخطوط المحيطة + النمط المنخفض ظل - رفع - + رفع + مزامنة الترجمة + 1000ms + تأخير الترجمة + استخدم هذا إذا كانت الترجمة تُعرض %dms مبكرًا جدًا + استخدم هذا إذا كانت الترجمة تُعرض %dms متأخرًا جدًا + لا تأخير في الترجمة + + نصٌّ حكيمٌ لهُ سِرٌّ قاطِعٌ وَذُو شَأنٍ عَظيمٍ مكتوبٌ على ثوبٍ أخضرَ ومُغلفٌ بجلدٍ أزرق - موصى به + مُوصي به + تم تحميل %s إختيار ملف + تحميل من الانترنت الملف الذي تم تنزيله رئيسي مساعد - خلفية - ترجمة كروم كاست - إعدادات ترجمة الكروم كاست - المصادر + الخلفية + + مصدر + عشوائي + + قريبا… + + Cam + Cam + Cam + HQ + HD + TS + TC + BlueRay + WP + DVD + 4K + SD + UHD + HDR + SDR + Web + + صورة الملصق + المُشغل + الأبعاد والعنوان + العنوان + الأبعاد + هوية غير صالحة + بيانات غير صالحة + عنون رابط غير صالح + خطأ + ازالة التسميات التوضيحية من الترجمة + ازالة البرمجيات الخبيثة من الترجمة + فلترة تبعا للغة المحتوي المفضلة + اكسترا + مقطع دعائي + رابط الفيديو + Referer + التالي + شاهد الفيديوهات بهذه اللغات + السابق + تخطي الإعداد + تغيير شكل البرنامح حتي يلائم جهازك + ابلاغ الاعطال + ماذا تريد ان تري + تم + الإضافات + إضافة مستودع + إسم المستودع + عنوان رابط المستودع + تم تحميل الإضافة + تم إزالة الإضافة + فشل التحميل %s + 18+ + بدأ تنزيل %d %s + تم تنزيل %d %s بنجاح + جميع %s محملة بالفعل + تحميل مكثف + إضافة + إضافات + سوف يتم محو جميع إضافات المستودع + ازالة مستودع + تحميل قائمة المواقع التي تريد استخدامها + تم تحميل: %d + مُعطل %d + غير مُحمل: %d + قم بإضافة مستودع لكي يتم تثبيت إضافات المواقع + عرض مستودعات المجتمع + قائمة عامة + جميع الترجمات حروف كبيرة + + تحميل جميع الإضافات من هذا المستودع? + %s (Disabled) + المسارات + مسار الصوت + مسار الفيديو + تطبيق بعد إعادة الفتح + + وضع الامان مفعل + An unrecoverable crash occurred and we\'ve automatically disabled all extensions, so you can find and remove the extension which is causing trouble. + عرض بيانات الاعطال + + تقييم: %s + الوصف + إصدار + الحالة + الحجم + المؤلفون + مدعوم + اللغة + + قائمة HLS From 9402a280413fb906d31bc6a3d28b1e585b2e7718 Mon Sep 17 00:00:00 2001 From: Hexated <37908684+hexated@users.noreply.github.com> Date: Wed, 14 Sep 2022 16:43:35 +0700 Subject: [PATCH 12/17] fixed StreamSB (#105) --- .../cloudstream3/extractors/Gdriveplayer.kt | 4 ++ .../cloudstream3/extractors/StreamSB.kt | 50 +++++++++++-------- .../cloudstream3/utils/ExtractorApi.kt | 2 + 3 files changed, 34 insertions(+), 22 deletions(-) diff --git a/app/src/main/java/com/lagradost/cloudstream3/extractors/Gdriveplayer.kt b/app/src/main/java/com/lagradost/cloudstream3/extractors/Gdriveplayer.kt index d2e56bf1..dfccc118 100644 --- a/app/src/main/java/com/lagradost/cloudstream3/extractors/Gdriveplayer.kt +++ b/app/src/main/java/com/lagradost/cloudstream3/extractors/Gdriveplayer.kt @@ -10,6 +10,10 @@ import javax.crypto.Cipher import javax.crypto.spec.IvParameterSpec import javax.crypto.spec.SecretKeySpec +class DatabaseGdrive : Gdriveplayer() { + override var mainUrl = "https://series.databasegdriveplayer.co" +} + class Gdriveplayerapi: Gdriveplayer() { override val mainUrl: String = "https://gdriveplayerapi.com" } diff --git a/app/src/main/java/com/lagradost/cloudstream3/extractors/StreamSB.kt b/app/src/main/java/com/lagradost/cloudstream3/extractors/StreamSB.kt index 461f56d0..30a0496d 100644 --- a/app/src/main/java/com/lagradost/cloudstream3/extractors/StreamSB.kt +++ b/app/src/main/java/com/lagradost/cloudstream3/extractors/StreamSB.kt @@ -1,12 +1,16 @@ package com.lagradost.cloudstream3.extractors import com.fasterxml.jackson.annotation.JsonProperty +import com.lagradost.cloudstream3.SubtitleFile import com.lagradost.cloudstream3.app -import com.lagradost.cloudstream3.utils.AppUtils.parseJson import com.lagradost.cloudstream3.utils.ExtractorApi import com.lagradost.cloudstream3.utils.ExtractorLink import com.lagradost.cloudstream3.utils.M3u8Helper +class Streamsss : StreamSB() { + override var mainUrl = "https://streamsss.net" +} + class Sbflix : StreamSB() { override var mainUrl = "https://sbflix.xyz" override var name = "Sbflix" @@ -109,31 +113,33 @@ open class StreamSB : ExtractorApi() { @JsonProperty("status_code") val statusCode: Int, ) - override suspend fun getUrl(url: String, referer: String?): List? { - val regexID = Regex("(embed-[a-zA-Z0-9]{0,8}[a-zA-Z0-9_-]+|\\/e\\/[a-zA-Z0-9]{0,8}[a-zA-Z0-9_-]+)") + override suspend fun getUrl( + url: String, + referer: String?, + subtitleCallback: (SubtitleFile) -> Unit, + callback: (ExtractorLink) -> Unit + ) { + val regexID = + Regex("(embed-[a-zA-Z0-9]{0,8}[a-zA-Z0-9_-]+|/e/[a-zA-Z0-9]{0,8}[a-zA-Z0-9_-]+)") val id = regexID.findAll(url).map { - it.value.replace(Regex("(embed-|\\/e\\/)"),"") + it.value.replace(Regex("(embed-|/e/)"), "") }.first() - val bytes = id.toByteArray() - val bytesToHex = bytesToHex(bytes) - val master = "$mainUrl/sources44/6d6144797752744a454267617c7c${bytesToHex.lowercase()}7c7c4e61755a56456f34385243727c7c73747265616d7362/6b4a33767968506e4e71374f7c7c343837323439333133333462353935333633373836643638376337633462333634663539343137373761333635313533333835333763376333393636363133393635366136323733343435323332376137633763373337343732363536313664373336327c7c504d754478413835306633797c7c73747265616d7362" +// val master = "$mainUrl/sources48/6d6144797752744a454267617c7c${bytesToHex.lowercase()}7c7c4e61755a56456f34385243727c7c73747265616d7362/6b4a33767968506e4e71374f7c7c343837323439333133333462353935333633373836643638376337633462333634663539343137373761333635313533333835333763376333393636363133393635366136323733343435323332376137633763373337343732363536313664373336327c7c504d754478413835306633797c7c73747265616d7362" + val master = "$mainUrl/sources48/" + bytesToHex("||$id||||streamsb".toByteArray()) + "/" val headers = mapOf( - "watchsb" to "streamsb", - ) - val urltext = app.get(master, + "watchsb" to "sbstream", + ) + val mapped = app.get( + master.lowercase(), headers = headers, - allowRedirects = false - ).text - val mapped = urltext.let { parseJson
(it) } - val testurl = app.get(mapped.streamData.file, headers = headers).text + referer = url, + ).parsedSafe
() // val urlmain = mapped.streamData.file.substringBefore("/hls/") - if (urltext.contains("m3u8") && testurl.contains("EXTM3U")) - return M3u8Helper.generateM3u8( - name, - mapped.streamData.file, - url, - headers = headers - ) - return null + M3u8Helper.generateM3u8( + name, + mapped?.streamData?.file ?: return, + url, + headers = headers + ).forEach(callback) } } \ No newline at end of file diff --git a/app/src/main/java/com/lagradost/cloudstream3/utils/ExtractorApi.kt b/app/src/main/java/com/lagradost/cloudstream3/utils/ExtractorApi.kt index ae3d8c06..b5c2cd44 100644 --- a/app/src/main/java/com/lagradost/cloudstream3/utils/ExtractorApi.kt +++ b/app/src/main/java/com/lagradost/cloudstream3/utils/ExtractorApi.kt @@ -237,6 +237,7 @@ val extractorApis: MutableList = arrayListOf( Sbthe(), Vidgomunime(), Sbflix(), + Streamsss(), Fastream(), @@ -333,6 +334,7 @@ val extractorApis: MutableList = arrayListOf( Gdriveplayerus(), Gdriveplayerco(), Gdriveplayer(), + DatabaseGdrive(), YoutubeExtractor(), YoutubeShortLinkExtractor(), From 0f492c2c82124742bd4dc3e4200a226597c4b5e2 Mon Sep 17 00:00:00 2001 From: Blatzar <46196380+Blatzar@users.noreply.github.com> Date: Sat, 17 Sep 2022 13:03:41 +0200 Subject: [PATCH 13/17] Added fallback webview to login pages and made account selectable on TV --- .../lagradost/cloudstream3/AcraApplication.kt | 31 ++++- .../com/lagradost/cloudstream3/MainAPI.kt | 1 + .../lagradost/cloudstream3/MainActivity.kt | 128 ++++++++++-------- .../cloudstream3/syncproviders/OAuth2API.kt | 4 +- .../syncproviders/providers/AniListApi.kt | 5 +- .../syncproviders/providers/DropboxApi.kt | 3 +- .../syncproviders/providers/MALApi.kt | 5 +- .../cloudstream3/ui/WebviewFragment.kt | 17 +-- .../ui/settings/SettingsAccount.kt | 40 +++--- .../lagradost/cloudstream3/utils/AppUtils.kt | 3 +- app/src/main/res/layout/main_settings.xml | 2 +- 11 files changed, 143 insertions(+), 96 deletions(-) diff --git a/app/src/main/java/com/lagradost/cloudstream3/AcraApplication.kt b/app/src/main/java/com/lagradost/cloudstream3/AcraApplication.kt index 85f031b3..198f0f4c 100644 --- a/app/src/main/java/com/lagradost/cloudstream3/AcraApplication.kt +++ b/app/src/main/java/com/lagradost/cloudstream3/AcraApplication.kt @@ -7,10 +7,12 @@ import android.content.ContextWrapper import android.content.Intent import android.widget.Toast import androidx.fragment.app.Fragment +import androidx.fragment.app.FragmentActivity import com.google.auto.service.AutoService import com.lagradost.cloudstream3.mvvm.normalSafeApiCall import com.lagradost.cloudstream3.mvvm.suspendSafeApiCall import com.lagradost.cloudstream3.plugins.PluginManager +import com.lagradost.cloudstream3.ui.settings.SettingsFragment.Companion.isTvSettings import com.lagradost.cloudstream3.utils.AppUtils.openBrowser import com.lagradost.cloudstream3.utils.Coroutines.runOnMainThread import com.lagradost.cloudstream3.utils.DataStore.getKey @@ -74,19 +76,28 @@ class CustomSenderFactory : ReportSenderFactory { } } -class ExceptionHandler(val errorFile: File, val onError: (() -> Unit)): Thread.UncaughtExceptionHandler { +class ExceptionHandler(val errorFile: File, val onError: (() -> Unit)) : + Thread.UncaughtExceptionHandler { override fun uncaughtException(thread: Thread, error: Throwable) { ACRA.errorReporter.handleException(error) try { PrintStream(errorFile).use { ps -> ps.println(String.format("Currently loading extension: ${PluginManager.currentlyLoading ?: "none"}")) - ps.println(String.format("Fatal exception on thread %s (%d)", thread.name, thread.id)) + ps.println( + String.format( + "Fatal exception on thread %s (%d)", + thread.name, + thread.id + ) + ) error.printStackTrace(ps) } - } catch (ignored: FileNotFoundException) { } + } catch (ignored: FileNotFoundException) { + } try { onError.invoke() - } catch (ignored: Exception) { } + } catch (ignored: Exception) { + } exitProcess(1) } @@ -95,7 +106,7 @@ class ExceptionHandler(val errorFile: File, val onError: (() -> Unit)): Thread.U class AcraApplication : Application() { override fun onCreate() { super.onCreate() - Thread.setDefaultUncaughtExceptionHandler(ExceptionHandler(filesDir.resolve("last_error")){ + Thread.setDefaultUncaughtExceptionHandler(ExceptionHandler(filesDir.resolve("last_error")) { val intent = context!!.packageManager.getLaunchIntentForPackage(context!!.packageName) startActivity(Intent.makeRestartActivityTask(intent!!.component)) }) @@ -183,5 +194,15 @@ class AcraApplication : Application() { fun openBrowser(url: String, fallbackWebview: Boolean = false, fragment: Fragment? = null) { context?.openBrowser(url, fallbackWebview, fragment) } + + /** Will fallback to webview if in TV layout */ + fun openBrowser(url: String, activity: FragmentActivity?) { + openBrowser( + url, + isTvSettings(), + activity?.supportFragmentManager?.fragments?.lastOrNull() + ) + } + } } \ No newline at end of file diff --git a/app/src/main/java/com/lagradost/cloudstream3/MainAPI.kt b/app/src/main/java/com/lagradost/cloudstream3/MainAPI.kt index d282f6dd..47afbc42 100644 --- a/app/src/main/java/com/lagradost/cloudstream3/MainAPI.kt +++ b/app/src/main/java/com/lagradost/cloudstream3/MainAPI.kt @@ -40,6 +40,7 @@ object APIHolder { private const val defProvider = 0 + // ConcurrentModificationException is possible!!! val allProviders: MutableList = arrayListOf() fun initAll() { diff --git a/app/src/main/java/com/lagradost/cloudstream3/MainActivity.kt b/app/src/main/java/com/lagradost/cloudstream3/MainActivity.kt index 6c9fadd8..7686a84c 100644 --- a/app/src/main/java/com/lagradost/cloudstream3/MainActivity.kt +++ b/app/src/main/java/com/lagradost/cloudstream3/MainActivity.kt @@ -15,6 +15,7 @@ import androidx.annotation.IdRes import androidx.appcompat.app.AlertDialog import androidx.appcompat.app.AppCompatActivity import androidx.core.view.isVisible +import androidx.fragment.app.FragmentActivity import androidx.navigation.NavController import androidx.navigation.NavDestination import androidx.navigation.NavDestination.Companion.hierarchy @@ -144,6 +145,68 @@ class MainActivity : AppCompatActivity(), ColorPickerDialogListener { val mainPluginsLoadedEvent = Event() // homepage api, used to speed up time to load for homepage val afterRepositoryLoadedEvent = Event() + + /** + * @return true if the str has launched an app task (be it successful or not) + * @param isWebview does not handle providers and opening download page if true. Can still add repos and login. + * */ + fun handleAppIntentUrl(activity: FragmentActivity?, str: String?, isWebview: Boolean): Boolean = + with(activity) { + if (str != null && this != null) { + if (str.startsWith("https://cs.repo")) { + val realUrl = "https://" + str.substringAfter("?") + println("Repository url: $realUrl") + loadRepository(realUrl) + return true + } else if (str.contains(appString)) { + for (api in OAuth2Apis) { + if (str.contains("/${api.redirectUrl}")) { + ioSafe { + Log.i(TAG, "handleAppIntent $str") + val isSuccessful = api.handleRedirect(str) + + if (isSuccessful) { + Log.i(TAG, "authenticated ${api.name}") + } else { + Log.i(TAG, "failed to authenticate ${api.name}") + } + + this@with.runOnUiThread { + try { + showToast( + this@with, + getString(if (isSuccessful) R.string.authenticated_user else R.string.authenticated_user_fail).format( + api.name + ) + ) + } catch (e: Exception) { + logError(e) // format might fail + } + } + } + return true + } + } + } else if (URI(str).scheme == appStringRepo) { + val url = str.replaceFirst(appStringRepo, "https") + loadRepository(url) + return true + } else if (!isWebview){ + if (str.startsWith(DOWNLOAD_NAVIGATE_TO)) { + this.navigate(R.id.navigation_downloads) + return true + } else { + for (api in apis) { + if (str.startsWith(api.mainUrl)) { + loadResult(str, api.name) + return true + } + } + } + } + } + return false + } } override fun onColorSelected(dialogId: Int, color: Int) { @@ -348,56 +411,7 @@ class MainActivity : AppCompatActivity(), ColorPickerDialogListener { if (intent == null) return val str = intent.dataString loadCache() - if (str != null) { - if (str.startsWith("https://cs.repo")) { - val realUrl = "https://" + str.substringAfter("?") - println("Repository url: $realUrl") - loadRepository(realUrl) - } else if (str.contains(appString)) { - for (api in OAuth2Apis) { - if (str.contains("/${api.redirectUrl}")) { - val activity = this - ioSafe { - Log.i(TAG, "handleAppIntent $str") - val isSuccessful = api.handleRedirect(str) - - if (isSuccessful) { - Log.i(TAG, "authenticated ${api.name}") - } else { - Log.i(TAG, "failed to authenticate ${api.name}") - } - - activity.runOnUiThread { - try { - showToast( - activity, - getString(if (isSuccessful) R.string.authenticated_user else R.string.authenticated_user_fail).format( - api.name - ) - ) - } catch (e: Exception) { - logError(e) // format might fail - } - } - } - } - } - } else if (URI(str).scheme == appStringRepo) { - val url = str.replaceFirst(appStringRepo, "https") - loadRepository(url) - } else { - if (str.startsWith(DOWNLOAD_NAVIGATE_TO)) { - this.navigate(R.id.navigation_downloads) - } else { - for (api in apis) { - if (str.startsWith(api.mainUrl)) { - loadResult(str, api.name) - break - } - } - } - } - } + handleAppIntentUrl(this, str, false) } private fun NavDestination.matchDestination(@IdRes destId: Int): Boolean = @@ -445,7 +459,8 @@ class MainActivity : AppCompatActivity(), ColorPickerDialogListener { } } // it.hashCode() is not enough to make sure they are distinct - apis = allProviders.distinctBy { it.lang + it.name + it.mainUrl + it.javaClass.name } + apis = + allProviders.distinctBy { it.lang + it.name + it.mainUrl + it.javaClass.name } APIHolder.apiMap = null } catch (e: Exception) { logError(e) @@ -465,9 +480,10 @@ class MainActivity : AppCompatActivity(), ColorPickerDialogListener { lastError = errorFile.readText(Charset.defaultCharset()) errorFile.delete() } - + val settingsForProvider = SettingsJson() - settingsForProvider.enableAdult = settingsManager.getBoolean(getString(R.string.enable_nsfw_on_providers_key), false) + settingsForProvider.enableAdult = + settingsManager.getBoolean(getString(R.string.enable_nsfw_on_providers_key), false) MainAPI.settingsForProvider = settingsForProvider @@ -501,7 +517,11 @@ class MainActivity : AppCompatActivity(), ColorPickerDialogListener { } ioSafe { - if (settingsManager.getBoolean(getString(R.string.auto_update_plugins_key), true)) { + if (settingsManager.getBoolean( + getString(R.string.auto_update_plugins_key), + true + ) + ) { PluginManager.updateAllOnlinePluginsAndLoadThem(this@MainActivity) } else { PluginManager.loadAllOnlinePlugins(this@MainActivity) diff --git a/app/src/main/java/com/lagradost/cloudstream3/syncproviders/OAuth2API.kt b/app/src/main/java/com/lagradost/cloudstream3/syncproviders/OAuth2API.kt index 0f882f3b..ef74edfc 100644 --- a/app/src/main/java/com/lagradost/cloudstream3/syncproviders/OAuth2API.kt +++ b/app/src/main/java/com/lagradost/cloudstream3/syncproviders/OAuth2API.kt @@ -1,9 +1,11 @@ package com.lagradost.cloudstream3.syncproviders +import androidx.fragment.app.FragmentActivity + interface OAuth2API : AuthAPI { val key: String val redirectUrl: String suspend fun handleRedirect(url: String) : Boolean - fun authenticate() + fun authenticate(activity: FragmentActivity?) } \ No newline at end of file diff --git a/app/src/main/java/com/lagradost/cloudstream3/syncproviders/providers/AniListApi.kt b/app/src/main/java/com/lagradost/cloudstream3/syncproviders/providers/AniListApi.kt index 606fee97..3140abbc 100644 --- a/app/src/main/java/com/lagradost/cloudstream3/syncproviders/providers/AniListApi.kt +++ b/app/src/main/java/com/lagradost/cloudstream3/syncproviders/providers/AniListApi.kt @@ -1,5 +1,6 @@ package com.lagradost.cloudstream3.syncproviders.providers +import androidx.fragment.app.FragmentActivity import com.fasterxml.jackson.annotation.JsonProperty import com.fasterxml.jackson.databind.DeserializationFeature import com.fasterxml.jackson.databind.json.JsonMapper @@ -48,9 +49,9 @@ class AniListApi(index: Int) : AccountManager(index), SyncAPI { removeAccountKeys() } - override fun authenticate() { + override fun authenticate(activity: FragmentActivity?) { val request = "https://anilist.co/api/v2/oauth/authorize?client_id=$key&response_type=token" - openBrowser(request) + openBrowser(request, activity) } override suspend fun handleRedirect(url: String): Boolean { diff --git a/app/src/main/java/com/lagradost/cloudstream3/syncproviders/providers/DropboxApi.kt b/app/src/main/java/com/lagradost/cloudstream3/syncproviders/providers/DropboxApi.kt index f847e0b2..7ec168da 100644 --- a/app/src/main/java/com/lagradost/cloudstream3/syncproviders/providers/DropboxApi.kt +++ b/app/src/main/java/com/lagradost/cloudstream3/syncproviders/providers/DropboxApi.kt @@ -1,5 +1,6 @@ package com.lagradost.cloudstream3.syncproviders.providers +import androidx.fragment.app.FragmentActivity import com.lagradost.cloudstream3.syncproviders.AuthAPI import com.lagradost.cloudstream3.syncproviders.OAuth2API @@ -15,7 +16,7 @@ class Dropbox : OAuth2API { override val icon: Int get() = TODO("Not yet implemented") - override fun authenticate() { + override fun authenticate(activity: FragmentActivity?) { TODO("Not yet implemented") } diff --git a/app/src/main/java/com/lagradost/cloudstream3/syncproviders/providers/MALApi.kt b/app/src/main/java/com/lagradost/cloudstream3/syncproviders/providers/MALApi.kt index ea27720a..c08958ce 100644 --- a/app/src/main/java/com/lagradost/cloudstream3/syncproviders/providers/MALApi.kt +++ b/app/src/main/java/com/lagradost/cloudstream3/syncproviders/providers/MALApi.kt @@ -1,6 +1,7 @@ package com.lagradost.cloudstream3.syncproviders.providers import android.util.Base64 +import androidx.fragment.app.FragmentActivity import com.fasterxml.jackson.annotation.JsonProperty import com.lagradost.cloudstream3.AcraApplication.Companion.getKey import com.lagradost.cloudstream3.AcraApplication.Companion.openBrowser @@ -281,7 +282,7 @@ class MALApi(index: Int) : AccountManager(index), SyncAPI { return false } - override fun authenticate() { + override fun authenticate(activity: FragmentActivity?) { // It is recommended to use a URL-safe string as code_verifier. // See section 4 of RFC 7636 for more details. @@ -294,7 +295,7 @@ class MALApi(index: Int) : AccountManager(index), SyncAPI { val codeChallenge = codeVerifier val request = "$mainUrl/v1/oauth2/authorize?response_type=code&client_id=$key&code_challenge=$codeChallenge&state=RequestID$requestId" - openBrowser(request) + openBrowser(request, activity) } private var requestId = 0 diff --git a/app/src/main/java/com/lagradost/cloudstream3/ui/WebviewFragment.kt b/app/src/main/java/com/lagradost/cloudstream3/ui/WebviewFragment.kt index cb4bbf37..fbc055a8 100644 --- a/app/src/main/java/com/lagradost/cloudstream3/ui/WebviewFragment.kt +++ b/app/src/main/java/com/lagradost/cloudstream3/ui/WebviewFragment.kt @@ -11,12 +11,12 @@ import android.webkit.WebViewClient import androidx.fragment.app.Fragment import androidx.fragment.app.FragmentActivity import androidx.navigation.fragment.findNavController +import com.lagradost.cloudstream3.MainActivity import com.lagradost.cloudstream3.R +import com.lagradost.cloudstream3.USER_AGENT import com.lagradost.cloudstream3.network.WebViewResolver -import com.lagradost.cloudstream3.syncproviders.AccountManager.Companion.appStringRepo import com.lagradost.cloudstream3.utils.AppUtils.loadRepository import kotlinx.android.synthetic.main.fragment_webview.* -import java.net.URI class WebviewFragment : Fragment() { override fun onViewCreated(view: View, savedInstanceState: Bundle?) { @@ -31,16 +31,8 @@ class WebviewFragment : Fragment() { request: WebResourceRequest? ): Boolean { val requestUrl = request?.url.toString() - val repoUrl = if (requestUrl.startsWith("https://cs.repo")) { - "https://" + requestUrl.substringAfter("?") - } else if (URI(requestUrl).scheme == appStringRepo) { - requestUrl.replaceFirst(appStringRepo, "https") - } else { - null - } - - if (repoUrl != null) { - activity?.loadRepository(repoUrl) + val performedAction = MainActivity.handleAppIntentUrl(activity, requestUrl, true) + if (performedAction) { findNavController().popBackStack() return true } @@ -50,6 +42,7 @@ class WebviewFragment : Fragment() { } web_view.addJavascriptInterface(RepoApi(activity), "RepoApi") web_view.settings.javaScriptEnabled = true + web_view.settings.userAgentString = USER_AGENT web_view.settings.domStorageEnabled = true WebViewResolver.webViewUserAgent = web_view.settings.userAgentString diff --git a/app/src/main/java/com/lagradost/cloudstream3/ui/settings/SettingsAccount.kt b/app/src/main/java/com/lagradost/cloudstream3/ui/settings/SettingsAccount.kt index 2554d6ee..f9627e46 100644 --- a/app/src/main/java/com/lagradost/cloudstream3/ui/settings/SettingsAccount.kt +++ b/app/src/main/java/com/lagradost/cloudstream3/ui/settings/SettingsAccount.kt @@ -1,8 +1,5 @@ package com.lagradost.cloudstream3.ui.settings -import android.app.Activity -import android.content.Intent -import android.net.Uri import android.os.Bundle import android.view.View import android.view.View.* @@ -12,8 +9,10 @@ import androidx.annotation.UiThread import androidx.appcompat.app.AlertDialog import androidx.core.view.isGone import androidx.core.view.isVisible +import androidx.fragment.app.FragmentActivity import androidx.preference.PreferenceFragmentCompat import androidx.recyclerview.widget.RecyclerView +import com.lagradost.cloudstream3.AcraApplication.Companion.openBrowser import com.lagradost.cloudstream3.CommonActivity.showToast import com.lagradost.cloudstream3.R import com.lagradost.cloudstream3.mvvm.logError @@ -39,7 +38,11 @@ import kotlinx.android.synthetic.main.add_account_input.* class SettingsAccount : PreferenceFragmentCompat() { companion object { /** Used by nginx plugin too */ - fun showLoginInfo(activity: Activity?, api: AccountManager, info: AuthAPI.LoginInfo) { + fun showLoginInfo( + activity: FragmentActivity?, + api: AccountManager, + info: AuthAPI.LoginInfo + ) { val builder = AlertDialog.Builder(activity ?: return, R.style.AlertDialogCustom) .setView(R.layout.account_managment) @@ -62,9 +65,13 @@ class SettingsAccount : PreferenceFragmentCompat() { dialog.dismissSafe(activity) showAccountSwitch(activity, api) } + + if (isTvSettings()) { + dialog.account_switch_account?.requestFocus() + } } - fun showAccountSwitch(activity: Activity, api: AccountManager) { + fun showAccountSwitch(activity: FragmentActivity, api: AccountManager) { val accounts = api.getAccounts() ?: return val builder = @@ -98,11 +105,11 @@ class SettingsAccount : PreferenceFragmentCompat() { } @UiThread - fun addAccount(activity: Activity?, api: AccountManager) { + fun addAccount(activity: FragmentActivity?, api: AccountManager) { try { when (api) { is OAuth2API -> { - api.authenticate() + api.authenticate(activity) } is InAppAuthAPI -> { val builder = @@ -144,13 +151,11 @@ class SettingsAccount : PreferenceFragmentCompat() { dialog.login_username_input?.isVisible = api.requiresUsername dialog.create_account?.isGone = api.createAccountUrl.isNullOrBlank() dialog.create_account?.setOnClickListener { - val i = Intent(Intent.ACTION_VIEW) - i.data = Uri.parse(api.createAccountUrl) - try { - activity.startActivity(i) - } catch (e: Exception) { - logError(e) - } + openBrowser( + api.createAccountUrl ?: return@setOnClickListener, + activity + ) + dialog.dismissSafe() } dialog.text1?.text = api.name @@ -181,9 +186,10 @@ class SettingsAccount : PreferenceFragmentCompat() { try { showToast( activity, - activity.getString(if (isSuccessful) R.string.authenticated_user else R.string.authenticated_user_fail).format( - api.name - ) + activity.getString(if (isSuccessful) R.string.authenticated_user else R.string.authenticated_user_fail) + .format( + api.name + ) ) } catch (e: Exception) { logError(e) // format might fail diff --git a/app/src/main/java/com/lagradost/cloudstream3/utils/AppUtils.kt b/app/src/main/java/com/lagradost/cloudstream3/utils/AppUtils.kt index 1c7bb214..cf3fbfde 100644 --- a/app/src/main/java/com/lagradost/cloudstream3/utils/AppUtils.kt +++ b/app/src/main/java/com/lagradost/cloudstream3/utils/AppUtils.kt @@ -31,6 +31,7 @@ import androidx.core.content.ContextCompat import androidx.core.text.HtmlCompat import androidx.core.text.toSpanned import androidx.fragment.app.Fragment +import androidx.fragment.app.FragmentActivity import androidx.navigation.fragment.findNavController import androidx.recyclerview.widget.LinearLayoutManager import androidx.recyclerview.widget.RecyclerView @@ -415,7 +416,7 @@ object AppUtils { } } - fun AppCompatActivity.loadResult( + fun FragmentActivity.loadResult( url: String, apiName: String, startAction: Int = 0, diff --git a/app/src/main/res/layout/main_settings.xml b/app/src/main/res/layout/main_settings.xml index 1527599c..387f98fa 100644 --- a/app/src/main/res/layout/main_settings.xml +++ b/app/src/main/res/layout/main_settings.xml @@ -101,7 +101,7 @@ Date: Sat, 17 Sep 2022 17:41:44 +0200 Subject: [PATCH 14/17] Update plugin backend --- .../cloudstream3/plugins/VotingApi.kt | 78 ++++++++++++------- 1 file changed, 48 insertions(+), 30 deletions(-) diff --git a/app/src/main/java/com/lagradost/cloudstream3/plugins/VotingApi.kt b/app/src/main/java/com/lagradost/cloudstream3/plugins/VotingApi.kt index ab702d71..f099ad1a 100644 --- a/app/src/main/java/com/lagradost/cloudstream3/plugins/VotingApi.kt +++ b/app/src/main/java/com/lagradost/cloudstream3/plugins/VotingApi.kt @@ -10,6 +10,9 @@ import java.security.MessageDigest import com.lagradost.cloudstream3.app import com.lagradost.cloudstream3.utils.Coroutines.ioSafe import com.lagradost.cloudstream3.utils.Coroutines.main +import kotlinx.coroutines.delay +import kotlinx.coroutines.sync.Mutex +import kotlinx.coroutines.sync.withLock object VotingApi { // please do not cheat the votes lol private const val LOGKEY = "VotingApi" @@ -23,10 +26,10 @@ object VotingApi { // please do not cheat the votes lol private val apiDomain = "https://api.countapi.xyz" private fun transformUrl(url: String): String = // dont touch or all votes get reset - MessageDigest - .getInstance("SHA-256") - .digest("${url}#funny-salt".toByteArray()) - .fold("") { str, it -> str + "%02x".format(it) } + MessageDigest + .getInstance("SHA-256") + .digest("${url}#funny-salt".toByteArray()) + .fold("") { str, it -> str + "%02x".format(it) } suspend fun SitePlugin.getVotes(): Int { return getVotes(url) @@ -53,9 +56,9 @@ object VotingApi { // please do not cheat the votes lol return votesCache[pluginUrl] ?: app.get(url).parsedSafe()?.value?.also { votesCache[pluginUrl] = it } ?: (0.also { - ioSafe { - createBucket(pluginUrl) - } + ioSafe { + createBucket(pluginUrl) + } }) } @@ -64,7 +67,8 @@ object VotingApi { // please do not cheat the votes lol } private suspend fun createBucket(pluginUrl: String) { - val url = "${apiDomain}/create?namespace=cs3-votes&key=${transformUrl(pluginUrl)}&value=0&update_lowerbound=-2&update_upperbound=2&enable_reset=0" + val url = + "${apiDomain}/create?namespace=cs3-votes&key=${transformUrl(pluginUrl)}&value=0&update_lowerbound=-2&update_upperbound=2&enable_reset=0" Log.d(LOGKEY, "Requesting: $url") app.get(url) } @@ -74,32 +78,46 @@ object VotingApi { // please do not cheat the votes lol return true } + private val voteLock = Mutex() suspend fun vote(pluginUrl: String, requestType: VoteType): Int { - if (!canVote(pluginUrl)) { - main { - Toast.makeText(context, R.string.extension_install_first, Toast.LENGTH_SHORT).show() + // Prevent multiple requests at the same time. + voteLock.withLock { + if (!canVote(pluginUrl)) { + main { + Toast.makeText(context, R.string.extension_install_first, Toast.LENGTH_SHORT) + .show() + } + return getVotes(pluginUrl) } - return getVotes(pluginUrl) - } - val savedType: VoteType = getKey("cs3-votes/${transformUrl(pluginUrl)}") ?: VoteType.NONE - var newType: VoteType = requestType - var changeValue = 0 - if (requestType == savedType) { - newType = VoteType.NONE - changeValue = -requestType.value - } else if (savedType == VoteType.NONE) { - changeValue = requestType.value - } else if (savedType != requestType) { - changeValue = -savedType.value + requestType.value - } - val url = "${apiDomain}/update/cs3-votes/${transformUrl(pluginUrl)}?amount=${changeValue}" - Log.d(LOGKEY, "Requesting: $url") - val res = app.get(url).parsedSafe()?.value - if (res != null) { + + val savedType: VoteType = + getKey("cs3-votes/${transformUrl(pluginUrl)}") ?: VoteType.NONE + + val newType = if (requestType == savedType) VoteType.NONE else requestType + val changeValue = if (requestType == savedType) { + -requestType.value + } else if (savedType == VoteType.NONE) { + requestType.value + } else if (savedType != requestType) { + -savedType.value + requestType.value + } else 0 + + // Pre-emptively set vote key setKey("cs3-votes/${transformUrl(pluginUrl)}", newType) - votesCache[pluginUrl] = res + + val url = + "${apiDomain}/update/cs3-votes/${transformUrl(pluginUrl)}?amount=${changeValue}" + Log.d(LOGKEY, "Requesting: $url") + val res = app.get(url).parsedSafe()?.value + + if (res == null) { + // "Refund" key if the response is invalid + setKey("cs3-votes/${transformUrl(pluginUrl)}", savedType) + } else { + votesCache[pluginUrl] = res + } + return res ?: 0 } - return res ?: 0 } private data class Result( From 9bbe3d65d2aed0b8a8686848ce4e20fb7d8f468e Mon Sep 17 00:00:00 2001 From: Cloudburst <18114966+C10udburst@users.noreply.github.com> Date: Sun, 18 Sep 2022 14:06:48 +0200 Subject: [PATCH 15/17] link dokka to github and make builds concurrent --- .../{issue-action.yml => issue_action.yml} | 126 +++++++++--------- .github/workflows/prerelease.yml | 4 +- app/build.gradle | 26 +++- 3 files changed, 86 insertions(+), 70 deletions(-) rename .github/workflows/{issue-action.yml => issue_action.yml} (97%) diff --git a/.github/workflows/issue-action.yml b/.github/workflows/issue_action.yml similarity index 97% rename from .github/workflows/issue-action.yml rename to .github/workflows/issue_action.yml index bfcb10d0..9ca9ff04 100644 --- a/.github/workflows/issue-action.yml +++ b/.github/workflows/issue_action.yml @@ -1,63 +1,63 @@ -name: Issue automatic actions - -on: - issues: - types: [opened, edited] - -jobs: - issue-moderator: - runs-on: ubuntu-latest - steps: - - name: Generate access token - id: generate_token - uses: tibdex/github-app-token@v1 - with: - app_id: ${{ secrets.GH_APP_ID }} - private_key: ${{ secrets.GH_APP_KEY }} - - name: Similarity analysis - uses: actions-cool/issues-similarity-analysis@v1 - with: - token: ${{ steps.generate_token.outputs.token }} - filter-threshold: 0.5 - title-excludes: '' - comment-title: | - ### Your issue looks similar to these issues: - Please close if duplicate. - comment-body: '${index}. ${similarity} #${number}' - - uses: actions/checkout@v2 - - name: Automatically close issues that dont follow the issue template - uses: lucasbento/auto-close-issues@v1.0.2 - with: - github-token: ${{ steps.generate_token.outputs.token }} - issue-close-message: | - @${issue.user.login}: hello! :wave: - This issue is being automatically closed because it does not follow the issue template." - closed-issues-label: "invalid" - - name: Check if issue mentions a provider - id: provider_check - env: - GH_TEXT: "${{ github.event.issue.title }} ${{ github.event.issue.body }}" - run: | - wget --output-document check_issue.py "https://raw.githubusercontent.com/recloudstream/.github/master/.github/check_issue.py" - pip3 install httpx - RES="$(python3 ./check_issue.py)" - echo "::set-output name=name::${RES}" - - name: Comment if issue mentions a provider - if: steps.provider_check.outputs.name != 'none' - uses: actions-cool/issues-helper@v3 - with: - actions: 'create-comment' - token: ${{ steps.generate_token.outputs.token }} - body: | - Hello ${{ github.event.issue.user.login }}. - Please do not report any provider bugs here. This repository does not contain any providers. Please find the appropriate repository and report your issue there or join the [discord](https://discord.gg/5Hus6fM). - - Found provider name: `${{ steps.provider_check.outputs.name }}` - - name: Add eyes reaction to all issues - uses: actions-cool/emoji-helper@v1.0.0 - with: - type: 'issue' - token: ${{ steps.generate_token.outputs.token }} - emoji: 'eyes' - - +name: Issue automatic actions + +on: + issues: + types: [opened, edited] + +jobs: + issue-moderator: + runs-on: ubuntu-latest + steps: + - name: Generate access token + id: generate_token + uses: tibdex/github-app-token@v1 + with: + app_id: ${{ secrets.GH_APP_ID }} + private_key: ${{ secrets.GH_APP_KEY }} + - name: Similarity analysis + uses: actions-cool/issues-similarity-analysis@v1 + with: + token: ${{ steps.generate_token.outputs.token }} + filter-threshold: 0.5 + title-excludes: '' + comment-title: | + ### Your issue looks similar to these issues: + Please close if duplicate. + comment-body: '${index}. ${similarity} #${number}' + - uses: actions/checkout@v2 + - name: Automatically close issues that dont follow the issue template + uses: lucasbento/auto-close-issues@v1.0.2 + with: + github-token: ${{ steps.generate_token.outputs.token }} + issue-close-message: | + @${issue.user.login}: hello! :wave: + This issue is being automatically closed because it does not follow the issue template." + closed-issues-label: "invalid" + - name: Check if issue mentions a provider + id: provider_check + env: + GH_TEXT: "${{ github.event.issue.title }} ${{ github.event.issue.body }}" + run: | + wget --output-document check_issue.py "https://raw.githubusercontent.com/recloudstream/.github/master/.github/check_issue.py" + pip3 install httpx + RES="$(python3 ./check_issue.py)" + echo "::set-output name=name::${RES}" + - name: Comment if issue mentions a provider + if: steps.provider_check.outputs.name != 'none' + uses: actions-cool/issues-helper@v3 + with: + actions: 'create-comment' + token: ${{ steps.generate_token.outputs.token }} + body: | + Hello ${{ github.event.issue.user.login }}. + Please do not report any provider bugs here. This repository does not contain any providers. Please find the appropriate repository and report your issue there or join the [discord](https://discord.gg/5Hus6fM). + + Found provider name: `${{ steps.provider_check.outputs.name }}` + - name: Add eyes reaction to all issues + uses: actions-cool/emoji-helper@v1.0.0 + with: + type: 'issue' + token: ${{ steps.generate_token.outputs.token }} + emoji: 'eyes' + + diff --git a/.github/workflows/prerelease.yml b/.github/workflows/prerelease.yml index 903878ed..37161d6b 100644 --- a/.github/workflows/prerelease.yml +++ b/.github/workflows/prerelease.yml @@ -43,9 +43,7 @@ jobs: echo "::set-output name=key_pwd::$KEY_PWD" - name: Run Gradle run: | - ./gradlew assemblePrerelease - ./gradlew androidSourcesJar - ./gradlew makeJar + ./gradlew assemblePrerelease makeJar androidSourcesJar env: SIGNING_KEY_ALIAS: "key0" SIGNING_KEY_PASSWORD: ${{ steps.fetch_keystore.outputs.key_pwd }} diff --git a/app/build.gradle b/app/build.gradle index b80c820f..291e71ff 100644 --- a/app/build.gradle +++ b/app/build.gradle @@ -206,10 +206,28 @@ task androidSourcesJar(type: Jar) { from android.sourceSets.main.java.srcDirs//full sources } +// this is used by the gradlew plugin task makeJar(type: Copy) { - // after modifying here, you can export. Jar from('build/intermediates/compile_app_classes_jar/debug') - into('build') // output location - include('classes.jar') // the classes file of the imported rack package - dependsOn build + into('build') + include('classes.jar') + dependsOn('build') } + +dokkaHtml { + moduleName.set("Cloudstream") + dokkaSourceSets { + main { + sourceLink { + // Unix based directory relative path to the root of the project (where you execute gradle respectively). + localDirectory.set(file("src/main/java")) + + // URL showing where the source code can be accessed through the web browser + remoteUrl.set(new URL( + "https://github.com/recloudstream/cloudstream/tree/master/app/src/main/java")) + // Suffix which is used to append the line number to the URL. Use #L for GitHub + remoteLineSuffix.set("#L") + } + } + } +} \ No newline at end of file From 29aa541dcee36506807cd8085d91922c18e79916 Mon Sep 17 00:00:00 2001 From: Blatzar <46196380+Blatzar@users.noreply.github.com> Date: Tue, 20 Sep 2022 21:53:26 +0200 Subject: [PATCH 16/17] Update Dutch translation to fix crashing caused by incorrect formatting --- app/src/main/res/values-nl/strings.xml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/app/src/main/res/values-nl/strings.xml b/app/src/main/res/values-nl/strings.xml index ae774f03..a5afd785 100644 --- a/app/src/main/res/values-nl/strings.xml +++ b/app/src/main/res/values-nl/strings.xml @@ -20,7 +20,7 @@ Snelheid (%.2fx) - Beoordeeld: %.Als + Beoordeeld: %.1fAls Nieuwe update gevonden!\n%s -> %s Filler %d min From d057a733685df88e6d3efb52878649389b878fcf Mon Sep 17 00:00:00 2001 From: Cloudburst <18114966+C10udburst@users.noreply.github.com> Date: Tue, 20 Sep 2022 22:01:30 +0200 Subject: [PATCH 17/17] fix dokka? --- .github/workflows/generate_dokka.yml | 3 +-- crowdin.yml | 5 ----- 2 files changed, 1 insertion(+), 7 deletions(-) delete mode 100644 crowdin.yml diff --git a/.github/workflows/generate_dokka.yml b/.github/workflows/generate_dokka.yml index 032ea8d0..3c5caad7 100644 --- a/.github/workflows/generate_dokka.yml +++ b/.github/workflows/generate_dokka.yml @@ -39,9 +39,8 @@ jobs: - name: Clean old builds run: | - shopt -s extglob cd $GITHUB_WORKSPACE/dokka/ - rm -rf !(.git) + rm -rf "./-cloudstream" - name: Setup JDK 11 uses: actions/setup-java@v1 diff --git a/crowdin.yml b/crowdin.yml deleted file mode 100644 index 2cc75108..00000000 --- a/crowdin.yml +++ /dev/null @@ -1,5 +0,0 @@ -files: - - source: /app/src/main/res/values/strings.xml - translation: /app/src/main/res/values-%android_code%/strings.xml - - source: /app/src/main/res/values/array.xml - translation: /app/src/main/res/values-%android_code%/array.xml