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