From 8506f343567d0b0f99230898d1a6817f64b6b7b5 Mon Sep 17 00:00:00 2001 From: Zaw <42999156+ImZaw@users.noreply.github.com> Date: Wed, 4 Jan 2023 01:06:34 +0300 Subject: [PATCH] Added egydead, some changes on arabseed --- ArabSeedProvider/build.gradle.kts | 2 +- .../kotlin/com/arabseed/ArabSeedProvider.kt | 12 +- EgyDeadProvider/build.gradle.kts | 14 ++ EgyDeadProvider/src/main/AndroidManifest.xml | 2 + .../main/kotlin/com/egydead/EgyDeadPlugin.kt | 11 ++ .../kotlin/com/egydead/EgyDeadProvider.kt | 180 ++++++++++++++++++ 6 files changed, 209 insertions(+), 12 deletions(-) create mode 100644 EgyDeadProvider/build.gradle.kts create mode 100644 EgyDeadProvider/src/main/AndroidManifest.xml create mode 100644 EgyDeadProvider/src/main/kotlin/com/egydead/EgyDeadPlugin.kt create mode 100644 EgyDeadProvider/src/main/kotlin/com/egydead/EgyDeadProvider.kt diff --git a/ArabSeedProvider/build.gradle.kts b/ArabSeedProvider/build.gradle.kts index 04f9493..2b9098c 100644 --- a/ArabSeedProvider/build.gradle.kts +++ b/ArabSeedProvider/build.gradle.kts @@ -1,4 +1,4 @@ -version = 1 +version = 2 cloudstream { description = "" diff --git a/ArabSeedProvider/src/main/kotlin/com/arabseed/ArabSeedProvider.kt b/ArabSeedProvider/src/main/kotlin/com/arabseed/ArabSeedProvider.kt index a114bb4..40f16ab 100644 --- a/ArabSeedProvider/src/main/kotlin/com/arabseed/ArabSeedProvider.kt +++ b/ArabSeedProvider/src/main/kotlin/com/arabseed/ArabSeedProvider.kt @@ -1,16 +1,9 @@ package com.arabseed -import com.fasterxml.jackson.annotation.JsonProperty import com.lagradost.cloudstream3.* -import com.lagradost.cloudstream3.LoadResponse.Companion.addTrailer -import com.lagradost.cloudstream3.utils.AppUtils.parseJson import com.lagradost.cloudstream3.utils.ExtractorLink -import com.lagradost.cloudstream3.utils.Qualities import com.lagradost.cloudstream3.utils.loadExtractor import org.jsoup.nodes.Element -import java.net.URI -import java.util.* -import kotlin.collections.ArrayList class ArabSeed : MainAPI() { override var lang = "ar" @@ -155,10 +148,8 @@ class ArabSeed : MainAPI() { subtitleCallback: (SubtitleFile) -> Unit, callback: (ExtractorLink) -> Unit ): Boolean { - println(data) val doc = app.get(data).document val watchUrl = doc.select("a.watchBTn").attr("href") - println(watchUrl) val watchDoc = app.get(watchUrl, headers = mapOf("Referer" to mainUrl)).document val indexOperators = arrayListOf() val list: List = watchDoc.select("ul > li[data-link], ul > h3").mapIndexed { index, element -> @@ -177,7 +168,6 @@ class ArabSeed : MainAPI() { } else { watchLinks = arrayListOf(0 to list) } - println(watchLinks) watchLinks.apmap { (quality, links) -> links.apmap { val iframeUrl = it.attr("data-link") @@ -190,7 +180,7 @@ class ArabSeed : MainAPI() { "ArabSeed", sourceElement.attr("src"), data, - if(quality != 0) quality else it.text().replace("\\D".toRegex(),"").toInt(), + if(quality != 0) quality else it.text().replace(".*- ".toRegex(), "").replace("\\D".toRegex(),"").toInt(), !sourceElement.attr("type").contains("mp4") ) ) diff --git a/EgyDeadProvider/build.gradle.kts b/EgyDeadProvider/build.gradle.kts new file mode 100644 index 0000000..615fbc2 --- /dev/null +++ b/EgyDeadProvider/build.gradle.kts @@ -0,0 +1,14 @@ +version = 1 + +cloudstream { + description = "Some LinkBox links are broken idk why" + authors = listOf( "ImZaw" ) + + language = "ar" + + status = 1 + + tvTypes = listOf( "TvSeries" , "Movie", "Anime" ) + + iconUrl = "https://www.google.com/s2/favicons?domain=tv.egydead.live&sz=%size%" +} diff --git a/EgyDeadProvider/src/main/AndroidManifest.xml b/EgyDeadProvider/src/main/AndroidManifest.xml new file mode 100644 index 0000000..f497b01 --- /dev/null +++ b/EgyDeadProvider/src/main/AndroidManifest.xml @@ -0,0 +1,2 @@ + + \ No newline at end of file diff --git a/EgyDeadProvider/src/main/kotlin/com/egydead/EgyDeadPlugin.kt b/EgyDeadProvider/src/main/kotlin/com/egydead/EgyDeadPlugin.kt new file mode 100644 index 0000000..b159641 --- /dev/null +++ b/EgyDeadProvider/src/main/kotlin/com/egydead/EgyDeadPlugin.kt @@ -0,0 +1,11 @@ +package com.egydead +import com.lagradost.cloudstream3.plugins.CloudstreamPlugin +import com.lagradost.cloudstream3.plugins.Plugin +import android.content.Context + +@CloudstreamPlugin +class EgyDeadPlugin: Plugin() { + override fun load(context: Context) { + registerMainAPI(EgyDead()) + } +} \ No newline at end of file diff --git a/EgyDeadProvider/src/main/kotlin/com/egydead/EgyDeadProvider.kt b/EgyDeadProvider/src/main/kotlin/com/egydead/EgyDeadProvider.kt new file mode 100644 index 0000000..ad0cb0a --- /dev/null +++ b/EgyDeadProvider/src/main/kotlin/com/egydead/EgyDeadProvider.kt @@ -0,0 +1,180 @@ +package com.egydead + +import com.fasterxml.jackson.annotation.JsonProperty +import com.lagradost.cloudstream3.* +import com.lagradost.cloudstream3.LoadResponse.Companion.addTrailer +import com.lagradost.cloudstream3.utils.ExtractorLink +import com.lagradost.cloudstream3.utils.Qualities +import com.lagradost.cloudstream3.utils.loadExtractor +import org.jsoup.nodes.Element +import java.net.URI + +class EgyDead : MainAPI() { + override var lang = "ar" + override var mainUrl = "https://tv.egydead.live" + override var name = "EgyDead" + override val usesWebView = false + override val hasMainPage = true + override val supportedTypes = setOf(TvType.TvSeries, TvType.Movie, TvType.Anime) + + private fun String.getIntFromText(): Int? { + return Regex("""\d+""").find(this)?.groupValues?.firstOrNull()?.toIntOrNull() + } + private fun String.cleanTitle(): String { + return this.replace("جميع مواسم مسلسل|مترجم كامل|مشاهدة فيلم|مترجم|انمي|الموسم.*|مترجمة كاملة|مسلسل|كاملة".toRegex(), "") + } + private fun Element.toSearchResponse(): SearchResponse { + val title = select("h1.BottomTitle").text().cleanTitle() + val posterUrl = select("img").attr("src") + val tvType = if (select("span.cat_name").text().contains("افلام")) TvType.Movie else TvType.TvSeries + return MovieSearchResponse( + title, + select("a").attr("href"), + this@EgyDead.name, + tvType, + posterUrl, + ) + } + + override val mainPage = mainPageOf( + "$mainUrl/category/افلام-اجنبي/?page=" to "English Movies", + "$mainUrl/category/افلام-اسيوية/?page=" to "Asian Movies", + "$mainUrl/season/?page=" to "Series", + ) + + override suspend fun getMainPage( + page: Int, + request: MainPageRequest + ): HomePageResponse { + val document = app.get(request.data + page).document + val home = document.select("li.movieItem").mapNotNull { + it.toSearchResponse() + } + return newHomePageResponse(request.name, home) + } + + override suspend fun search(query: String): List { + val doc = app.get("$mainUrl/?s=$query").document + return doc.select("li.movieItem").mapNotNull { + if(it.select("a").attr("href").contains("/episode/|/season/".toRegex())) return@mapNotNull null + it.toSearchResponse() + } + } + + override suspend fun load(url: String): LoadResponse { + val doc = app.get(url).document + val title = doc.select("div.singleTitle em").text().cleanTitle() + val isMovie = !url.contains("/serie/") + + val posterUrl = doc.select("div.single-thumbnail > img").attr("src") + val rating = doc.select("a.IMDBRating em").text().getIntFromText() + val synopsis = doc.select("div.extra-content:contains(القصه) p").text() + val year = doc.select("ul > li:contains(السنه) > a").text().getIntFromText() + val tags = doc.select("ul > li:contains(النوع) > a").map { it.text() } + val recommendations = doc.select("div.related-posts > ul > li").mapNotNull { element -> + element.toSearchResponse() + } + val youtubeTrailer = doc.select("div.popupContent > iframe").attr("src") + return if (isMovie) { + newMovieLoadResponse( + title, + url, + TvType.Movie, + url + ) { + this.posterUrl = posterUrl + this.recommendations = recommendations + this.plot = synopsis + this.tags = tags + this.rating = rating + this.year = year + addTrailer(youtubeTrailer) + } + } else { + val seasonList = doc.select("div.seasons-list ul > li > a").reversed() + val episodes = arrayListOf() + if(seasonList.isNotEmpty()) { + seasonList.apmapIndexed { index, season -> + app.get( + season.attr("href"), + ).document.select("div.EpsList > li > a").map { + episodes.add(Episode( + it.attr("href"), + it.attr("title"), + index+1, + it.text().getIntFromText() + )) + } + } + } else { + doc.select("div.EpsList > li > a").map { + episodes.add(Episode( + it.attr("href"), + it.attr("title"), + 0, + it.text().getIntFromText() + )) + } + } + newTvSeriesLoadResponse(title, url, TvType.TvSeries, episodes.distinct().sortedBy { it.episode }) { + this.posterUrl = posterUrl + this.tags = tags + this.plot = synopsis + this.recommendations = recommendations + this.rating = rating + this.year = year + addTrailer(youtubeTrailer) + } + } + } + data class LinkBox ( + @JsonProperty("data" ) var data : Data? = Data(), + ) + data class Data ( + @JsonProperty("rList" ) var rList : ArrayList = arrayListOf(), + ) + data class RList ( + @JsonProperty("resolution" ) var resolution : String? = null, + @JsonProperty("size" ) var size : Double? = null, +// @JsonProperty("sub_type" ) var subType : String? = null, + @JsonProperty("url" ) var url : String? = null, + + ) + private fun bytesToHumanReadableSize(bytes: Double) = when { + bytes >= 1 shl 30 -> "%.1f GB".format(bytes / (1 shl 30)) + bytes >= 1 shl 20 -> "%.1f MB".format(bytes / (1 shl 20)) + bytes >= 1 shl 10 -> "%.0f kB".format(bytes / (1 shl 10)) + else -> "$bytes bytes" + } + override suspend fun loadLinks( + data: String, + isCasting: Boolean, + subtitleCallback: (SubtitleFile) -> Unit, + callback: (ExtractorLink) -> Unit + ): Boolean { + val doc = app.post(data, data = mapOf("View" to "1")).document + val watchList = doc.select("ul.serversList > li") + watchList.apmap { li -> + val iframeUrl = li.attr("data-link") + val quality = li.select("small").text().getIntFromText() ?: Qualities.Unknown.value + if(iframeUrl.contains("www.linkbox.to")) { + val apiUrl = "https://" + URI(iframeUrl).host + "/api/open/get_url?itemId=" + iframeUrl.substringAfter("id=").substringBefore("&") + val json = app.get(apiUrl).parsed() + json.data?.rList?.forEach { + callback.invoke( + ExtractorLink( + this.name, + "LinkBox " + bytesToHumanReadableSize(it.size ?: 0.0), + it.url ?: return@forEach, + mainUrl, + quality, + false + ) + ) + } + } + else loadExtractor(iframeUrl, data, subtitleCallback, callback) + } + return true + } +} \ No newline at end of file