From 80c026e8981d79a6691999e5e6a729b45465b3b2 Mon Sep 17 00:00:00 2001 From: C10udburst <18114966+C10udburst@users.noreply.github.com> Date: Wed, 17 Aug 2022 16:46:03 +0200 Subject: [PATCH] add zaluknij.xyz --- ZaluknijProvider/build.gradle.kts | 26 ++++ ZaluknijProvider/src/main/AndroidManifest.xml | 2 + .../kotlin/com/lagradost/ZaluknijProvider.kt | 141 ++++++++++++++++++ .../com/lagradost/ZaluknijProviderPlugin.kt | 13 ++ 4 files changed, 182 insertions(+) create mode 100644 ZaluknijProvider/build.gradle.kts create mode 100644 ZaluknijProvider/src/main/AndroidManifest.xml create mode 100644 ZaluknijProvider/src/main/kotlin/com/lagradost/ZaluknijProvider.kt create mode 100644 ZaluknijProvider/src/main/kotlin/com/lagradost/ZaluknijProviderPlugin.kt diff --git a/ZaluknijProvider/build.gradle.kts b/ZaluknijProvider/build.gradle.kts new file mode 100644 index 0000000..2866981 --- /dev/null +++ b/ZaluknijProvider/build.gradle.kts @@ -0,0 +1,26 @@ +// use an integer for version numbers +version = 1 + + +cloudstream { + language = "pl" + // All of these properties are optional, you can safely remove them + + // description = "Lorem Ipsum" + authors = listOf("Cloudburst") + + /** + * Status int as the following: + * 0: Down + * 1: Ok + * 2: Slow + * 3: Beta only + * */ + status = 1 + tvTypes = listOf( + "Movie", + "TvSeries" + ) + + iconUrl = "https://www.google.com/s2/favicons?domain=zaluknij.xyz&sz=24" +} diff --git a/ZaluknijProvider/src/main/AndroidManifest.xml b/ZaluknijProvider/src/main/AndroidManifest.xml new file mode 100644 index 0000000..29aec9d --- /dev/null +++ b/ZaluknijProvider/src/main/AndroidManifest.xml @@ -0,0 +1,2 @@ + + \ No newline at end of file diff --git a/ZaluknijProvider/src/main/kotlin/com/lagradost/ZaluknijProvider.kt b/ZaluknijProvider/src/main/kotlin/com/lagradost/ZaluknijProvider.kt new file mode 100644 index 0000000..2a37c73 --- /dev/null +++ b/ZaluknijProvider/src/main/kotlin/com/lagradost/ZaluknijProvider.kt @@ -0,0 +1,141 @@ +package com.lagradost + +import com.fasterxml.jackson.annotation.JsonProperty +import com.lagradost.cloudstream3.* +import com.lagradost.cloudstream3.utils.AppUtils.tryParseJson +import com.lagradost.cloudstream3.utils.ExtractorLink +import com.lagradost.cloudstream3.utils.loadExtractor +import org.jsoup.Jsoup +import org.jsoup.select.Elements + +class ZaluknijProvider : MainAPI() { + override var mainUrl = "https://zaluknij.xyz/" + override var name = "zaluknij.xyz" + override var lang = "pl" + override val hasMainPage = true + override val usesWebView = true + override val supportedTypes = setOf( + TvType.TvSeries, + TvType.Movie + ) + + override suspend fun getMainPage(page: Int, request : MainPageRequest): HomePageResponse { + val document = app.get(mainUrl).document + val lists = document.select(".item-list") + val categories = ArrayList() + for (l in lists) { + val title = capitalizeString(l.parent()!!.select("h3").text().lowercase().trim()) + val items = l.select(".poster").map { i -> + val a = i.parent()!! + val name = a.attr("title") + val href = a.attr("href") + val poster = i.select("img[src]").attr("src") + val year = a.select(".year").text().toIntOrNull() + MovieSearchResponse( + name, + href, + this.name, + TvType.Movie, + poster, + year + ) + } + categories.add(HomePageList(title, items)) + } + return HomePageResponse(categories) + } + + override suspend fun search(query: String): List { + val url = "$mainUrl/wyszukiwarka?phrase=$query" + val document = app.get(url).document + val lists = document.select("#advanced-search > div") + val movies = lists[1].select("div:not(.clearfix)") + val series = lists[3].select("div:not(.clearfix)") + if (movies.isEmpty() && series.isEmpty()) return ArrayList() + fun getVideos(type: TvType, items: Elements): List { + return items.mapNotNull { i -> + val href = i.selectFirst("a")?.attr("href") ?: return@mapNotNull null + val img = + i.selectFirst("a > img[src]")?.attr("src")?.replace("/thumb/", "/big/") + val name = i.selectFirst(".title")?.text() ?: return@mapNotNull null + if (type === TvType.TvSeries) { + TvSeriesSearchResponse( + name, + href, + this.name, + type, + img, + null, + null + ) + } else { + MovieSearchResponse(name, href, this.name, type, img, null) + } + } + } + return getVideos(TvType.Movie, movies) + getVideos(TvType.TvSeries, series) + } + + override suspend fun load(url: String): LoadResponse { + val document = app.get(url).document + val documentTitle = document.select("title").text().trim() + + if (documentTitle.startsWith("Logowanie")) { + throw RuntimeException("This page seems to be locked behind a login-wall on the website, unable to scrape it. If it is not please report it.") + } + + var title = document.select("span[itemprop=name]").text() + val data = document.select("#link-list").outerHtml() + val posterUrl = document.select("#single-poster > img").attr("src") + val plot = document.select(".description").text() + val episodesElements = document.select("#episode-list a[href]") + if (episodesElements.isEmpty()) { + return MovieLoadResponse(title, url, name, TvType.Movie, data, posterUrl, null, plot) + } + title = document.selectFirst(".info")?.parent()?.select("h2")?.text()!! + val episodes = episodesElements.mapNotNull { episode -> + val e = episode.text() + val regex = Regex("""\[s(\d{1,3})e(\d{1,3})]""").find(e) ?: return@mapNotNull null + val eid = regex.groups + Episode( + episode.attr("href"), + e.split("]")[1].trim(), + eid[1]?.value?.toInt(), + eid[2]?.value?.toInt(), + ) + }.toMutableList() + + return TvSeriesLoadResponse( + title, + url, + name, + TvType.TvSeries, + episodes, + posterUrl, + null, + plot + ) + } + + override suspend fun loadLinks( + data: String, + isCasting: Boolean, + subtitleCallback: (SubtitleFile) -> Unit, + callback: (ExtractorLink) -> Unit + ): Boolean { + val document = if (data.startsWith("http")) + app.get(data).document.select("#link-list").first() + else Jsoup.parse(data) + + document?.select(".link-to-video")?.apmap { item -> + val decoded = base64Decode(item.select("a").attr("data-iframe")) + val link = tryParseJson(decoded)?.src ?: return@apmap + loadExtractor(link, subtitleCallback, callback) + } + return true + } +} + +data class LinkElement( + @JsonProperty("src") val src: String +) diff --git a/ZaluknijProvider/src/main/kotlin/com/lagradost/ZaluknijProviderPlugin.kt b/ZaluknijProvider/src/main/kotlin/com/lagradost/ZaluknijProviderPlugin.kt new file mode 100644 index 0000000..f966aa7 --- /dev/null +++ b/ZaluknijProvider/src/main/kotlin/com/lagradost/ZaluknijProviderPlugin.kt @@ -0,0 +1,13 @@ + +package com.lagradost + +import com.lagradost.cloudstream3.plugins.CloudstreamPlugin +import com.lagradost.cloudstream3.plugins.Plugin +import android.content.Context + +@CloudstreamPlugin +class ZaluknijProviderPlugin: Plugin() { + override fun load(context: Context) { + registerMainAPI(ZaluknijProvider()) + } +} \ No newline at end of file