diff --git a/AltadefinizioneProvider/build.gradle.kts b/AltadefinizioneProvider/build.gradle.kts
new file mode 100644
index 0000000..549ae23
--- /dev/null
+++ b/AltadefinizioneProvider/build.gradle.kts
@@ -0,0 +1,24 @@
+// use an integer for version numbers
+version = 1
+
+
+cloudstream {
+ // 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 // will be 3 if unspecified
+ tvTypes = listOf(
+ "Movie",
+ )
+
+ iconUrl = "https://www.google.com/s2/favicons?domain=altadefinizione.tienda&sz=24"
+}
\ No newline at end of file
diff --git a/AltadefinizioneProvider/src/main/AndroidManifest.xml b/AltadefinizioneProvider/src/main/AndroidManifest.xml
new file mode 100644
index 0000000..29aec9d
--- /dev/null
+++ b/AltadefinizioneProvider/src/main/AndroidManifest.xml
@@ -0,0 +1,2 @@
+
+
\ No newline at end of file
diff --git a/AltadefinizioneProvider/src/main/kotlin/com/lagradost/AltadefinizioneProvider.kt b/AltadefinizioneProvider/src/main/kotlin/com/lagradost/AltadefinizioneProvider.kt
new file mode 100644
index 0000000..1a3fa77
--- /dev/null
+++ b/AltadefinizioneProvider/src/main/kotlin/com/lagradost/AltadefinizioneProvider.kt
@@ -0,0 +1,160 @@
+package com.lagradost
+
+//import androidx.core.text.parseAsHtml
+import com.lagradost.cloudstream3.*
+import com.lagradost.cloudstream3.LoadResponse.Companion.addTrailer
+import com.lagradost.cloudstream3.utils.ExtractorLink
+import com.lagradost.cloudstream3.utils.loadExtractor
+import com.lagradost.cloudstream3.utils.AppUtils.html
+
+
+class AltadefinizioneProvider : MainAPI() {
+ override var lang = "it"
+ override var mainUrl = "https://altadefinizione.tienda"
+ override var name = "Altadefinizione"
+ override val hasMainPage = true
+ override val hasChromecastSupport = true
+ override val supportedTypes = setOf(
+ TvType.Movie
+ )
+
+ override val mainPage = mainPageOf(
+ Pair("$mainUrl/cerca/anno/2022/page/", "Ultimi Film"),
+ Pair("$mainUrl/cerca/openload-quality/HD/page/", "Film in HD"),
+ Pair("$mainUrl/cinema/page/", "Ora al cinema")
+ )
+
+ override suspend fun getMainPage(page: Int, request: MainPageRequest): HomePageResponse {
+ val url = request.data + page
+
+ val soup = app.get(url).document
+ val home = soup.select("div.box").map {
+ val title = it.selectFirst("img")!!.attr("alt")
+ val link = it.selectFirst("a")!!.attr("href")
+ val image = mainUrl + it.selectFirst("img")!!.attr("src")
+ val quality = getQualityFromString(it.selectFirst("span")!!.text())
+
+ MovieSearchResponse(
+ title,
+ link,
+ this.name,
+ TvType.Movie,
+ image,
+ null,
+ null,
+ quality,
+ )
+ }
+ return newHomePageResponse(request.name, home)
+ }
+
+ override suspend fun search(query: String): List {
+ val doc = app.post(
+ "$mainUrl/index.php", data = mapOf(
+ "do" to "search",
+ "subaction" to "search",
+ "story" to query,
+ "sortby" to "news_read"
+ )
+ ).document
+ return doc.select("div.box").map {
+ val title = it.selectFirst("img")!!.attr("alt")
+ val link = it.selectFirst("a")!!.attr("href")
+ val image = mainUrl + it.selectFirst("img")!!.attr("src")
+ val quality = getQualityFromString(it.selectFirst("span")!!.text())
+
+ MovieSearchResponse(
+ title,
+ link,
+ this.name,
+ TvType.Movie,
+ image,
+ null,
+ null,
+ quality,
+ )
+ }
+ }
+
+ override suspend fun load(url: String): LoadResponse {
+ val page = app.get(url)
+ val document = page.document
+ val title = document.selectFirst(" h1 > a")!!.text().replace("streaming", "")
+ val description = document.select("#sfull").toString().substringAfter("altadefinizione")
+ .substringBeforeLast("fonte trama").html().toString()
+ val rating = null
+
+ val year = document.selectFirst("#details > li:nth-child(2)")!!.childNode(2).toString()
+ .filter { it.isDigit() }.toInt()
+
+ val poster = fixUrl(document.selectFirst("div.thumbphoto > img")!!.attr("src"))
+
+ val recomm = document.select("ul.related-list > li").map {
+ val href = it.selectFirst("a")!!.attr("href")
+ val posterUrl = mainUrl + it.selectFirst("img")!!.attr("src")
+ val name = it.selectFirst("img")!!.attr("alt")
+ MovieSearchResponse(
+ name,
+ href,
+ this.name,
+ TvType.Movie,
+ posterUrl,
+ null
+ )
+
+ }
+
+
+ val actors: List =
+ document.select("#staring > a").map {
+ ActorData(actor = Actor(it.text()))
+ }
+
+ val tags: List = document.select("#details > li:nth-child(1) > a").map { it.text() }
+
+ val trailerurl = document.selectFirst("#showtrailer > div > div > iframe")?.attr("src")
+
+ return newMovieLoadResponse(
+ title,
+ url,
+ TvType.Movie,
+ url
+ ) {
+ posterUrl = fixUrlNull(poster)
+ this.year = year
+ this.plot = description
+ this.rating = rating
+ this.recommendations = recomm
+ this.duration = null
+ this.actors = actors
+ this.tags = tags
+ addTrailer(trailerurl)
+ }
+ }
+
+
+ override suspend fun loadLinks(
+ data: String,
+ isCasting: Boolean,
+ subtitleCallback: (SubtitleFile) -> Unit,
+ callback: (ExtractorLink) -> Unit
+ ): Boolean {
+ val doc = app.get(data).document
+ if (doc.select("div.guardahd-player").isNullOrEmpty()) {
+ val videoUrl =
+ doc.select("input").last { it.hasAttr("data-mirror") }.attr("value")
+ loadExtractor(videoUrl, data, subtitleCallback, callback)
+ doc.select("#mirrors > li > a").forEach {
+ loadExtractor(fixUrl(it.attr("data-target")), data, subtitleCallback, callback)
+ }
+ } else {
+ val pagelinks = doc.select("div.guardahd-player").select("iframe").attr("src")
+ val docLinks = app.get(pagelinks).document
+ docLinks.select("body > div > ul > li").forEach {
+ loadExtractor(fixUrl(it.attr("data-link")), data, subtitleCallback, callback)
+ }
+ }
+
+ return true
+ }
+}
\ No newline at end of file
diff --git a/AltadefinizioneProvider/src/main/kotlin/com/lagradost/AltadefinizioneProviderPlugin.kt b/AltadefinizioneProvider/src/main/kotlin/com/lagradost/AltadefinizioneProviderPlugin.kt
new file mode 100644
index 0000000..1e9a49b
--- /dev/null
+++ b/AltadefinizioneProvider/src/main/kotlin/com/lagradost/AltadefinizioneProviderPlugin.kt
@@ -0,0 +1,14 @@
+
+package com.lagradost
+
+import com.lagradost.cloudstream3.plugins.CloudstreamPlugin
+import com.lagradost.cloudstream3.plugins.Plugin
+import android.content.Context
+
+@CloudstreamPlugin
+class AltadefinizioneProviderPlugin: Plugin() {
+ override fun load(context: Context) {
+ // All providers should be added in this manner. Please don't edit the providers list directly.
+ registerMainAPI(AltadefinizioneProvider())
+ }
+}
\ No newline at end of file
diff --git a/AniPlayProvider/build.gradle.kts b/AniPlayProvider/build.gradle.kts
new file mode 100644
index 0000000..0684a8b
--- /dev/null
+++ b/AniPlayProvider/build.gradle.kts
@@ -0,0 +1,27 @@
+// use an integer for version numbers
+version = 1
+
+
+cloudstream {
+ // 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 // will be 3 if unspecified
+ tvTypes = listOf(
+ "AnimeMovie",
+ "Anime",
+ "OVA"
+ )
+
+
+ iconUrl = "https://www.google.com/s2/favicons?domain=aniplay.it&sz=24"
+}
\ No newline at end of file
diff --git a/AniPlayProvider/src/main/AndroidManifest.xml b/AniPlayProvider/src/main/AndroidManifest.xml
new file mode 100644
index 0000000..29aec9d
--- /dev/null
+++ b/AniPlayProvider/src/main/AndroidManifest.xml
@@ -0,0 +1,2 @@
+
+
\ No newline at end of file
diff --git a/AniPlayProvider/src/main/kotlin/com/lagradost/AniPlayProvider.kt b/AniPlayProvider/src/main/kotlin/com/lagradost/AniPlayProvider.kt
new file mode 100644
index 0000000..15eb860
--- /dev/null
+++ b/AniPlayProvider/src/main/kotlin/com/lagradost/AniPlayProvider.kt
@@ -0,0 +1,215 @@
+package com.lagradost
+
+import com.fasterxml.jackson.annotation.JsonProperty
+import com.lagradost.cloudstream3.*
+import com.lagradost.cloudstream3.LoadResponse.Companion.addAniListId
+import com.lagradost.cloudstream3.LoadResponse.Companion.addDuration
+import com.lagradost.cloudstream3.LoadResponse.Companion.addMalId
+import com.lagradost.cloudstream3.utils.ExtractorLink
+import com.lagradost.cloudstream3.utils.M3u8Helper
+import com.lagradost.cloudstream3.utils.Qualities
+
+class AniPlayProvider : MainAPI() {
+ override var mainUrl = "https://aniplay.it"
+ override var name = "AniPlay"
+ override var lang = "it"
+ override val hasMainPage = true
+ private val dubIdentifier = " (ITA)"
+
+ override val supportedTypes = setOf(
+ TvType.Anime,
+ TvType.AnimeMovie,
+ TvType.OVA
+ )
+
+ companion object {
+ fun getStatus(t: String?): ShowStatus? {
+ return when (t?.lowercase()) {
+ "completato" -> ShowStatus.Completed
+ "in corso" -> ShowStatus.Ongoing
+ else -> null // "annunciato"
+ }
+ }
+ fun getType(t: String?): TvType {
+ return when (t?.lowercase()) {
+ "ona" -> TvType.OVA
+ "movie" -> TvType.AnimeMovie
+ else -> TvType.Anime //"serie", "special"
+ }
+ }
+ }
+
+ private fun isDub(title: String): Boolean{
+ return title.contains(dubIdentifier)
+ }
+
+ data class ApiPoster(
+ @JsonProperty("imageFull") val posterUrl: String
+ )
+
+ data class ApiMainPageAnime(
+ @JsonProperty("animeId") val id: Int,
+ @JsonProperty("episodeNumber") val episode: String?,
+ @JsonProperty("animeTitle") val title: String,
+ @JsonProperty("animeType") val type: String,
+ @JsonProperty("fullHd") val fullHD: Boolean,
+ @JsonProperty("animeVerticalImages") val posters: List
+ )
+
+ data class ApiSearchResult(
+ @JsonProperty("id") val id: Int,
+ @JsonProperty("title") val title: String,
+ @JsonProperty("status") val status: String,
+ @JsonProperty("type") val type: String,
+ @JsonProperty("verticalImages") val posters: List
+ )
+
+ data class ApiGenres(
+ @JsonProperty("description") val name: String
+ )
+ data class ApiWebsite(
+ @JsonProperty("listWebsiteId") val websiteId: Int,
+ @JsonProperty("url") val url: String
+ )
+
+ data class ApiEpisode(
+ @JsonProperty("id") val id: Int,
+ @JsonProperty("title") val title: String?,
+ @JsonProperty("episodeNumber") val number: String,
+ )
+
+ private fun ApiEpisode.toEpisode() : Episode? {
+ val number = this.number.toIntOrNull() ?: return null
+ return Episode(
+ data = "$mainUrl/api/episode/${this.id}",
+ episode = number,
+ name = this.title
+ )
+ }
+
+ data class ApiSeason(
+ @JsonProperty("id") val id: Int,
+ @JsonProperty("name") val name: String
+ )
+
+ private suspend fun ApiSeason.toEpisodeList(url: String) : List {
+ return app.get("$url/season/${this.id}").parsed>().mapNotNull { it.toEpisode() }
+ }
+
+ data class ApiAnime(
+ @JsonProperty("title") val title: String,
+ @JsonProperty("alternativeTitle") val japTitle: String?,
+ @JsonProperty("episodeDuration") val duration: Int,
+ @JsonProperty("storyline") val plot: String,
+ @JsonProperty("type") val type: String,
+ @JsonProperty("status") val status: String,
+ @JsonProperty("genres") val genres: List,
+ @JsonProperty("verticalImages") val posters: List,
+ @JsonProperty("listWebsites") val websites: List,
+ @JsonProperty("episodes") val episodes: List,
+ @JsonProperty("seasons") val seasons: List?
+ )
+
+ data class ApiEpisodeUrl(
+ @JsonProperty("videoUrl") val url: String
+ )
+
+ override suspend fun getMainPage(page: Int, request : MainPageRequest): HomePageResponse {
+ val response = app.get("$mainUrl/api/home/latest-episodes?page=0").parsed>()
+
+ val results = response.map{
+ val isDub = isDub(it.title)
+ newAnimeSearchResponse(
+ name = if (isDub) it.title.replace(dubIdentifier, "") else it.title,
+ url = "$mainUrl/api/anime/${it.id}",
+ type = getType(it.type),
+ ){
+ addDubStatus(isDub, it.episode?.toIntOrNull())
+ this.posterUrl = it.posters.first().posterUrl
+ this.quality = if (it.fullHD) SearchQuality.HD else null
+ }
+ }
+ return HomePageResponse(listOf(HomePageList("Ultime uscite",results)))
+ }
+
+ override suspend fun search(query: String): List {
+ val response = app.get("$mainUrl/api/anime/advanced-search?page=0&size=36&query=$query").parsed>()
+
+ return response.map {
+ val isDub = isDub(it.title)
+
+ newAnimeSearchResponse(
+ name = if (isDub) it.title.replace(dubIdentifier, "") else it.title,
+ url = "$mainUrl/api/anime/${it.id}",
+ type = getType(it.type),
+ ){
+ addDubStatus(isDub)
+ this.posterUrl = it.posters.first().posterUrl
+ }
+ }
+ }
+
+ override suspend fun load(url: String): LoadResponse {
+
+ val response = app.get(url).parsed()
+
+ val tags: List = response.genres.map { it.name }
+
+ val malId: Int? = response.websites.find { it.websiteId == 1 }?.url?.removePrefix("https://myanimelist.net/anime/")?.split("/")?.first()?.toIntOrNull()
+ val aniListId: Int? = response.websites.find { it.websiteId == 4 }?.url?.removePrefix("https://anilist.co/anime/")?.split("/")?.first()?.toIntOrNull()
+
+ val episodes = if (response.seasons.isNullOrEmpty()) response.episodes.mapNotNull { it.toEpisode() } else response.seasons.map{ it.toEpisodeList(url) }.flatten()
+ val isDub = isDub(response.title)
+
+ return newAnimeLoadResponse(response.title, url, getType(response.type)) {
+ this.name = if (isDub) response.title.replace(dubIdentifier, "") else response.title
+ this.japName = response.japTitle
+ this.plot = response.plot
+ this.tags = tags
+ this.showStatus = getStatus(response.status)
+ addPoster(response.posters.first().posterUrl)
+ addEpisodes(if (isDub) DubStatus.Dubbed else DubStatus.Subbed, episodes)
+ addMalId(malId)
+ addAniListId(aniListId)
+ addDuration(response.duration.toString())
+ }
+ }
+
+ override suspend fun loadLinks(
+ data: String,
+ isCasting: Boolean,
+ subtitleCallback: (SubtitleFile) -> Unit,
+ callback: (ExtractorLink) -> Unit
+ ): Boolean {
+
+ val episode = app.get(data).parsed()
+
+ if(episode.url.contains(".m3u8")){
+ val m3u8Helper = M3u8Helper()
+ val streams = m3u8Helper.m3u8Generation(M3u8Helper.M3u8Stream(episode.url,Qualities.Unknown.value), false)
+
+ streams.forEach {
+ callback.invoke(
+ ExtractorLink(
+ name,
+ name,
+ it.streamUrl,
+ referer = mainUrl,
+ quality = it.quality ?: Qualities.Unknown.value,
+ isM3u8 = it.streamUrl.contains(".m3u8"))) }
+ return true
+ }
+
+ callback.invoke(
+ ExtractorLink(
+ name,
+ name,
+ episode.url,
+ referer = mainUrl,
+ quality = Qualities.Unknown.value,
+ isM3u8 = false,
+ )
+ )
+ return true
+ }
+}
\ No newline at end of file
diff --git a/AniPlayProvider/src/main/kotlin/com/lagradost/AniPlayProviderPlugin.kt b/AniPlayProvider/src/main/kotlin/com/lagradost/AniPlayProviderPlugin.kt
new file mode 100644
index 0000000..de3dccb
--- /dev/null
+++ b/AniPlayProvider/src/main/kotlin/com/lagradost/AniPlayProviderPlugin.kt
@@ -0,0 +1,14 @@
+
+package com.lagradost
+
+import com.lagradost.cloudstream3.plugins.CloudstreamPlugin
+import com.lagradost.cloudstream3.plugins.Plugin
+import android.content.Context
+
+@CloudstreamPlugin
+class AniPlayProviderPlugin: Plugin() {
+ override fun load(context: Context) {
+ // All providers should be added in this manner. Please don't edit the providers list directly.
+ registerMainAPI(AniPlayProvider())
+ }
+}
\ No newline at end of file
diff --git a/AnimeSaturnProvider/build.gradle.kts b/AnimeSaturnProvider/build.gradle.kts
new file mode 100644
index 0000000..70a6c1a
--- /dev/null
+++ b/AnimeSaturnProvider/build.gradle.kts
@@ -0,0 +1,25 @@
+// use an integer for version numbers
+version = 1
+
+
+cloudstream {
+ // 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 // will be 3 if unspecified
+ tvTypes = listOf(
+ "Anime",
+ "AnimeMovie",
+ )
+
+ iconUrl = "https://www.google.com/s2/favicons?domain=www.animesaturn.cc&sz=24"
+}
\ No newline at end of file
diff --git a/AnimeSaturnProvider/src/main/AndroidManifest.xml b/AnimeSaturnProvider/src/main/AndroidManifest.xml
new file mode 100644
index 0000000..29aec9d
--- /dev/null
+++ b/AnimeSaturnProvider/src/main/AndroidManifest.xml
@@ -0,0 +1,2 @@
+
+
\ No newline at end of file
diff --git a/AnimeSaturnProvider/src/main/kotlin/com/lagradost/AnimeSaturnProvider.kt b/AnimeSaturnProvider/src/main/kotlin/com/lagradost/AnimeSaturnProvider.kt
new file mode 100644
index 0000000..dadb2b5
--- /dev/null
+++ b/AnimeSaturnProvider/src/main/kotlin/com/lagradost/AnimeSaturnProvider.kt
@@ -0,0 +1,201 @@
+package com.lagradost
+
+import com.lagradost.cloudstream3.*
+import com.lagradost.cloudstream3.LoadResponse.Companion.addAniListId
+import com.lagradost.cloudstream3.LoadResponse.Companion.addDuration
+import com.lagradost.cloudstream3.LoadResponse.Companion.addMalId
+import com.lagradost.cloudstream3.LoadResponse.Companion.addRating
+import com.lagradost.cloudstream3.utils.ExtractorLink
+import com.lagradost.cloudstream3.utils.Qualities
+import org.jsoup.nodes.Element
+
+class AnimeSaturnProvider : MainAPI() {
+ override var mainUrl = "https://www.animesaturn.cc"
+ override var name = "AnimeSaturn"
+ override var lang = "it"
+ override val hasMainPage = true
+
+ override val supportedTypes = setOf(
+ TvType.Anime,
+ TvType.AnimeMovie,
+ TvType.OVA
+ )
+
+ companion object {
+ fun getStatus(t: String?): ShowStatus? {
+ return when (t?.lowercase()) {
+ "finito" -> ShowStatus.Completed
+ "in corso" -> ShowStatus.Ongoing
+ else -> null
+ }
+ }
+ }
+
+ private fun Element.toSearchResult(): AnimeSearchResponse {
+
+ var title = this.select("a.badge-archivio").first()!!.text()
+ var isDubbed = false
+
+ if (title.contains(" (ITA)")){
+ title = title.replace(" (ITA)", "")
+ isDubbed = true
+ }
+
+ val url = this.select("a.badge-archivio").first()!!.attr("href")
+
+ val posterUrl = this.select("img.locandina-archivio[src]").first()!!.attr("src")
+
+ return newAnimeSearchResponse(title, url, TvType.Anime) {
+ addDubStatus(isDubbed)
+ this.posterUrl = posterUrl
+ }
+ }
+
+ private fun Element.toEpisode(): Episode? {
+ var episode = this.text().split(" ")[1]
+ if(episode.contains(".")) return null
+ if(episode.contains("-"))
+ episode = episode.split("-")[0]
+
+ return Episode(
+ data = this.attr("href"),
+ episode = episode.toInt()
+ )
+
+ }
+
+ override suspend fun getMainPage(page: Int, request : MainPageRequest): HomePageResponse {
+ val document = app.get(mainUrl).document
+ val list = ArrayList()
+ document.select("div.container:has(span.badge-saturn)").forEach {
+ val tabName = it.select("span.badge-saturn").first()!!.text()
+ if (tabName.equals("Ultimi episodi")) return@forEach
+ val results = ArrayList()
+ it.select(".main-anime-card").forEach { card ->
+ var title = card.select("a[title]").first()!!.attr("title")
+ var isDubbed = false
+ if(title.contains(" (ITA)")){
+ title = title.replace(" (ITA)", "")
+ isDubbed = true
+ }
+ val posterUrl = card.select("img.new-anime").first()!!.attr("src")
+ val url = card.select("a").first()!!.attr("href")
+
+ results.add(newAnimeSearchResponse(title, url, TvType.Anime){
+ addDubStatus(isDubbed)
+ this.posterUrl = posterUrl
+ })
+ }
+ list.add(HomePageList(tabName, results))
+ }
+ return HomePageResponse(list)
+ }
+
+ override suspend fun search(query: String): List {
+ val document = app.get("$mainUrl/animelist?search=$query").document
+ return document.select("div.item-archivio").map {
+ it.toSearchResult()
+ }
+ }
+
+ override suspend fun load(url: String): LoadResponse {
+
+ val document = app.get(url).document
+
+ val title = document.select("img.cover-anime").first()!!.attr("alt")
+ val japTitle = document.select("div.box-trasparente-alternativo").first()!!.text()
+ val posterUrl = document.select("img.cover-anime[src]").first()!!.attr("src")
+ var malId : Int? = null
+ var aniListId : Int? = null
+
+ document.select("[rel=\"noopener noreferrer\"]").forEach {
+ if(it.attr("href").contains("myanimelist"))
+ malId = it.attr("href").removeSuffix("/").split('/').last().toIntOrNull()
+ else
+ aniListId = it.attr("href").removeSuffix("/").split('/').last().toIntOrNull()
+ }
+
+ val plot = document.select("div#shown-trama").first()?.text()
+
+ val tags = document.select("a.generi-as").map { it.text() }
+
+ val details : List? = document.select("div.container:contains(Stato: )").first()?.text()?.split(" ")
+ var status : String? = null
+ var duration : String? = null
+ var year : String? = null
+ var score : String? = null
+
+ val isDubbed = document.select("div.anime-title-as").first()!!.text().contains("(ITA)")
+
+ if (!details.isNullOrEmpty()) {
+ details.forEach {
+ val index = details.indexOf(it) +1
+ when (it) {
+ "Stato:" -> status = details[index]
+ "episodi:" -> duration = details[index]
+ "uscita:" -> year = details[index + 2]
+ "Voto:" -> score = details[index].split("/")[0]
+ else -> return@forEach
+ }
+ }
+ }
+
+ val episodes = document.select("a.bottone-ep").mapNotNull{ it.toEpisode() }
+
+ return newAnimeLoadResponse(title, url, TvType.Anime) {
+ this.engName = title
+ this.japName = japTitle
+ this.year = year?.toIntOrNull()
+ this.plot = plot
+ this.tags = tags
+ this.showStatus = getStatus(status)
+ addPoster(posterUrl)
+ addRating(score)
+ addEpisodes(if (isDubbed) DubStatus.Dubbed else DubStatus.Subbed, episodes)
+ addMalId(malId)
+ addAniListId(aniListId)
+ addDuration(duration)
+ }
+ }
+
+ override suspend fun loadLinks(
+ data: String,
+ isCasting: Boolean,
+ subtitleCallback: (SubtitleFile) -> Unit,
+ callback: (ExtractorLink) -> Unit
+ ): Boolean {
+
+ val page = app.get(data).document
+ val episodeLink = page.select("div.card-body > a[href]").find {it1 ->
+ it1.attr("href").contains("watch?")
+ }?.attr("href")
+
+ val episodePage = app.get(episodeLink!!).document
+ val episodeUrl: String?
+ var isM3U8 = false
+
+ if(episodePage.select("video.afterglow > source").isNotEmpty()) //Old player
+ episodeUrl = episodePage.select("video.afterglow > source").first()!!.attr("src")
+
+ else{ //New player
+ val script = episodePage.select("script").find {
+ it.toString().contains("jwplayer('player_hls').setup({")
+ }!!.toString()
+ episodeUrl = script.split(" ").find { it.contains(".m3u8") and !it.contains(".replace") }!!.replace("\"","").replace(",", "")
+ isM3U8 = true
+ }
+
+
+ callback.invoke(
+ ExtractorLink(
+ name,
+ name,
+ episodeUrl!!,
+ isM3u8 = isM3U8,
+ referer = "https://www.animesaturn.io/", //Some servers need the old host as referer, and the new ones accept it too
+ quality = Qualities.Unknown.value
+ )
+ )
+ return true
+ }
+}
diff --git a/AnimeSaturnProvider/src/main/kotlin/com/lagradost/AnimeSaturnProviderPlugin.kt b/AnimeSaturnProvider/src/main/kotlin/com/lagradost/AnimeSaturnProviderPlugin.kt
new file mode 100644
index 0000000..048741d
--- /dev/null
+++ b/AnimeSaturnProvider/src/main/kotlin/com/lagradost/AnimeSaturnProviderPlugin.kt
@@ -0,0 +1,14 @@
+
+package com.lagradost
+
+import com.lagradost.cloudstream3.plugins.CloudstreamPlugin
+import com.lagradost.cloudstream3.plugins.Plugin
+import android.content.Context
+
+@CloudstreamPlugin
+class AnimeSaturnProviderPlugin: Plugin() {
+ override fun load(context: Context) {
+ // All providers should be added in this manner. Please don't edit the providers list directly.
+ registerMainAPI(AnimeSaturnProvider())
+ }
+}
\ No newline at end of file
diff --git a/AnimeWorldProvider/build.gradle.kts b/AnimeWorldProvider/build.gradle.kts
new file mode 100644
index 0000000..6454bba
--- /dev/null
+++ b/AnimeWorldProvider/build.gradle.kts
@@ -0,0 +1,26 @@
+// use an integer for version numbers
+version = 1
+
+
+cloudstream {
+ // 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 // will be 3 if unspecified
+ tvTypes = listOf(
+ "AnimeMovie",
+ "Anime",
+ "OVA",
+ )
+
+ iconUrl = "https://www.google.com/s2/favicons?domain=www.animeworld.tv&sz=24"
+}
\ No newline at end of file
diff --git a/AnimeWorldProvider/src/main/AndroidManifest.xml b/AnimeWorldProvider/src/main/AndroidManifest.xml
new file mode 100644
index 0000000..29aec9d
--- /dev/null
+++ b/AnimeWorldProvider/src/main/AndroidManifest.xml
@@ -0,0 +1,2 @@
+
+
\ No newline at end of file
diff --git a/AnimeWorldProvider/src/main/kotlin/com/lagradost/AnimeWorldProvider.kt b/AnimeWorldProvider/src/main/kotlin/com/lagradost/AnimeWorldProvider.kt
new file mode 100644
index 0000000..3d454cb
--- /dev/null
+++ b/AnimeWorldProvider/src/main/kotlin/com/lagradost/AnimeWorldProvider.kt
@@ -0,0 +1,265 @@
+package com.lagradost
+
+import com.fasterxml.jackson.annotation.JsonProperty
+import com.lagradost.cloudstream3.*
+import com.lagradost.cloudstream3.LoadResponse.Companion.addAniListId
+import com.lagradost.cloudstream3.LoadResponse.Companion.addDuration
+import com.lagradost.cloudstream3.LoadResponse.Companion.addMalId
+import com.lagradost.cloudstream3.LoadResponse.Companion.addRating
+import com.lagradost.cloudstream3.LoadResponse.Companion.addTrailer
+import com.lagradost.cloudstream3.utils.AppUtils.tryParseJson
+import com.lagradost.cloudstream3.utils.ExtractorLink
+import com.lagradost.cloudstream3.utils.Qualities
+import com.lagradost.nicehttp.NiceResponse
+import org.jsoup.nodes.Element
+import org.mozilla.javascript.ConsString
+import org.mozilla.javascript.Context
+import org.mozilla.javascript.Scriptable
+
+class AnimeWorldProvider : MainAPI() {
+ override var mainUrl = "https://www.animeworld.tv"
+ override var name = "AnimeWorld"
+ override var lang = "it"
+ override val hasMainPage = true
+
+ override val supportedTypes = setOf(
+ TvType.Anime,
+ TvType.AnimeMovie,
+ TvType.OVA
+ )
+
+ companion object {
+ private var cookies = emptyMap()
+
+ // Disabled authentication as site did
+ private suspend fun request(url: String): NiceResponse {
+// if (cookies.isEmpty()) {
+// cookies = getCookies(url)
+// }
+ return app.get(url
+// , cookies = cookies
+ )
+ }
+
+ private suspend fun getCookies(url: String): Map {
+ val rhino = Context.enter()
+ rhino.optimizationLevel = -1
+ val scope: Scriptable = rhino.initSafeStandardObjects()
+
+ val slowAes = app.get("https://www.animeworld.tv/aes.min.js").text
+// val slowAes = """
+// var _0x2465=["\x72\x6F\x74\x61\x74\x65","\x73\x62\x6F\x78","\x52\x63\x6F\x6E","\x6E\x75\x6D\x62\x65\x72\x4F\x66\x52\x6F\x75\x6E\x64\x73","\x63\x6F\x72\x65","\x53\x49\x5A\x45\x5F\x32\x35\x36","\x6B\x65\x79\x53\x69\x7A\x65","\x72\x73\x62\x6F\x78","\x73\x68\x69\x66\x74\x52\x6F\x77","\x6D\x69\x78\x43\x6F\x6C\x75\x6D\x6E","\x67\x61\x6C\x6F\x69\x73\x5F\x6D\x75\x6C\x74\x69\x70\x6C\x69\x63\x61\x74\x69\x6F\x6E","\x73\x75\x62\x42\x79\x74\x65\x73","\x73\x68\x69\x66\x74\x52\x6F\x77\x73","\x6D\x69\x78\x43\x6F\x6C\x75\x6D\x6E\x73","\x61\x64\x64\x52\x6F\x75\x6E\x64\x4B\x65\x79","\x63\x72\x65\x61\x74\x65\x52\x6F\x75\x6E\x64\x4B\x65\x79","\x72\x6F\x75\x6E\x64","\x69\x6E\x76\x52\x6F\x75\x6E\x64","\x53\x49\x5A\x45\x5F\x31\x32\x38","\x53\x49\x5A\x45\x5F\x31\x39\x32","\x65\x78\x70\x61\x6E\x64\x4B\x65\x79","\x6D\x61\x69\x6E","\x69\x6E\x76\x4D\x61\x69\x6E","\x73\x6C\x69\x63\x65","\x6C\x65\x6E\x67\x74\x68","\x69\x76\x20\x6C\x65\x6E\x67\x74\x68\x20\x6D\x75\x73\x74\x20\x62\x65\x20\x31\x32\x38\x20\x62\x69\x74\x73\x2E","\x43\x42\x43","\x6D\x6F\x64\x65\x4F\x66\x4F\x70\x65\x72\x61\x74\x69\x6F\x6E","\x70\x61\x64\x42\x79\x74\x65\x73\x49\x6E","\x63\x65\x69\x6C","\x67\x65\x74\x42\x6C\x6F\x63\x6B","\x43\x46\x42","\x65\x6E\x63\x72\x79\x70\x74","\x61\x65\x73","\x70\x75\x73\x68","\x4F\x46\x42","\x64\x65\x63\x72\x79\x70\x74","\x75\x6E\x70\x61\x64\x42\x79\x74\x65\x73\x4F\x75\x74","\x73\x70\x6C\x69\x63\x65"];var slowAES={aes:{keySize:{SIZE_128:16,SIZE_192:24,SIZE_256:32},sbox:[0x63,0x7c,0x77,0x7b,0xf2,0x6b,0x6f,0xc5,0x30,0x01,0x67,0x2b,0xfe,0xd7,0xab,0x76,0xca,0x82,0xc9,0x7d,0xfa,0x59,0x47,0xf0,0xad,0xd4,0xa2,0xaf,0x9c,0xa4,0x72,0xc0,0xb7,0xfd,0x93,0x26,0x36,0x3f,0xf7,0xcc,0x34,0xa5,0xe5,0xf1,0x71,0xd8,0x31,0x15,0x04,0xc7,0x23,0xc3,0x18,0x96,0x05,0x9a,0x07,0x12,0x80,0xe2,0xeb,0x27,0xb2,0x75,0x09,0x83,0x2c,0x1a,0x1b,0x6e,0x5a,0xa0,0x52,0x3b,0xd6,0xb3,0x29,0xe3,0x2f,0x84,0x53,0xd1,0x00,0xed,0x20,0xfc,0xb1,0x5b,0x6a,0xcb,0xbe,0x39,0x4a,0x4c,0x58,0xcf,0xd0,0xef,0xaa,0xfb,0x43,0x4d,0x33,0x85,0x45,0xf9,0x02,0x7f,0x50,0x3c,0x9f,0xa8,0x51,0xa3,0x40,0x8f,0x92,0x9d,0x38,0xf5,0xbc,0xb6,0xda,0x21,0x10,0xff,0xf3,0xd2,0xcd,0x0c,0x13,0xec,0x5f,0x97,0x44,0x17,0xc4,0xa7,0x7e,0x3d,0x64,0x5d,0x19,0x73,0x60,0x81,0x4f,0xdc,0x22,0x2a,0x90,0x88,0x46,0xee,0xb8,0x14,0xde,0x5e,0x0b,0xdb,0xe0,0x32,0x3a,0x0a,0x49,0x06,0x24,0x5c,0xc2,0xd3,0xac,0x62,0x91,0x95,0xe4,0x79,0xe7,0xc8,0x37,0x6d,0x8d,0xd5,0x4e,0xa9,0x6c,0x56,0xf4,0xea,0x65,0x7a,0xae,0x08,0xba,0x78,0x25,0x2e,0x1c,0xa6,0xb4,0xc6,0xe8,0xdd,0x74,0x1f,0x4b,0xbd,0x8b,0x8a,0x70,0x3e,0xb5,0x66,0x48,0x03,0xf6,0x0e,0x61,0x35,0x57,0xb9,0x86,0xc1,0x1d,0x9e,0xe1,0xf8,0x98,0x11,0x69,0xd9,0x8e,0x94,0x9b,0x1e,0x87,0xe9,0xce,0x55,0x28,0xdf,0x8c,0xa1,0x89,0x0d,0xbf,0xe6,0x42,0x68,0x41,0x99,0x2d,0x0f,0xb0,0x54,0xbb,0x16],rsbox:[0x52,0x09,0x6a,0xd5,0x30,0x36,0xa5,0x38,0xbf,0x40,0xa3,0x9e,0x81,0xf3,0xd7,0xfb,0x7c,0xe3,0x39,0x82,0x9b,0x2f,0xff,0x87,0x34,0x8e,0x43,0x44,0xc4,0xde,0xe9,0xcb,0x54,0x7b,0x94,0x32,0xa6,0xc2,0x23,0x3d,0xee,0x4c,0x95,0x0b,0x42,0xfa,0xc3,0x4e,0x08,0x2e,0xa1,0x66,0x28,0xd9,0x24,0xb2,0x76,0x5b,0xa2,0x49,0x6d,0x8b,0xd1,0x25,0x72,0xf8,0xf6,0x64,0x86,0x68,0x98,0x16,0xd4,0xa4,0x5c,0xcc,0x5d,0x65,0xb6,0x92,0x6c,0x70,0x48,0x50,0xfd,0xed,0xb9,0xda,0x5e,0x15,0x46,0x57,0xa7,0x8d,0x9d,0x84,0x90,0xd8,0xab,0x00,0x8c,0xbc,0xd3,0x0a,0xf7,0xe4,0x58,0x05,0xb8,0xb3,0x45,0x06,0xd0,0x2c,0x1e,0x8f,0xca,0x3f,0x0f,0x02,0xc1,0xaf,0xbd,0x03,0x01,0x13,0x8a,0x6b,0x3a,0x91,0x11,0x41,0x4f,0x67,0xdc,0xea,0x97,0xf2,0xcf,0xce,0xf0,0xb4,0xe6,0x73,0x96,0xac,0x74,0x22,0xe7,0xad,0x35,0x85,0xe2,0xf9,0x37,0xe8,0x1c,0x75,0xdf,0x6e,0x47,0xf1,0x1a,0x71,0x1d,0x29,0xc5,0x89,0x6f,0xb7,0x62,0x0e,0xaa,0x18,0xbe,0x1b,0xfc,0x56,0x3e,0x4b,0xc6,0xd2,0x79,0x20,0x9a,0xdb,0xc0,0xfe,0x78,0xcd,0x5a,0xf4,0x1f,0xdd,0xa8,0x33,0x88,0x07,0xc7,0x31,0xb1,0x12,0x10,0x59,0x27,0x80,0xec,0x5f,0x60,0x51,0x7f,0xa9,0x19,0xb5,0x4a,0x0d,0x2d,0xe5,0x7a,0x9f,0x93,0xc9,0x9c,0xef,0xa0,0xe0,0x3b,0x4d,0xae,0x2a,0xf5,0xb0,0xc8,0xeb,0xbb,0x3c,0x83,0x53,0x99,0x61,0x17,0x2b,0x04,0x7e,0xba,0x77,0xd6,0x26,0xe1,0x69,0x14,0x63,0x55,0x21,0x0c,0x7d],rotate:function(_0x85f6x2){var _0x85f6x3=_0x85f6x2[0];for(var _0x85f6x4=0;_0x85f6x4< 3;_0x85f6x4++){_0x85f6x2[_0x85f6x4]= _0x85f6x2[_0x85f6x4+ 1]};_0x85f6x2[3]= _0x85f6x3;return _0x85f6x2},Rcon:[0x8d,0x01,0x02,0x04,0x08,0x10,0x20,0x40,0x80,0x1b,0x36,0x6c,0xd8,0xab,0x4d,0x9a,0x2f,0x5e,0xbc,0x63,0xc6,0x97,0x35,0x6a,0xd4,0xb3,0x7d,0xfa,0xef,0xc5,0x91,0x39,0x72,0xe4,0xd3,0xbd,0x61,0xc2,0x9f,0x25,0x4a,0x94,0x33,0x66,0xcc,0x83,0x1d,0x3a,0x74,0xe8,0xcb,0x8d,0x01,0x02,0x04,0x08,0x10,0x20,0x40,0x80,0x1b,0x36,0x6c,0xd8,0xab,0x4d,0x9a,0x2f,0x5e,0xbc,0x63,0xc6,0x97,0x35,0x6a,0xd4,0xb3,0x7d,0xfa,0xef,0xc5,0x91,0x39,0x72,0xe4,0xd3,0xbd,0x61,0xc2,0x9f,0x25,0x4a,0x94,0x33,0x66,0xcc,0x83,0x1d,0x3a,0x74,0xe8,0xcb,0x8d,0x01,0x02,0x04,0x08,0x10,0x20,0x40,0x80,0x1b,0x36,0x6c,0xd8,0xab,0x4d,0x9a,0x2f,0x5e,0xbc,0x63,0xc6,0x97,0x35,0x6a,0xd4,0xb3,0x7d,0xfa,0xef,0xc5,0x91,0x39,0x72,0xe4,0xd3,0xbd,0x61,0xc2,0x9f,0x25,0x4a,0x94,0x33,0x66,0xcc,0x83,0x1d,0x3a,0x74,0xe8,0xcb,0x8d,0x01,0x02,0x04,0x08,0x10,0x20,0x40,0x80,0x1b,0x36,0x6c,0xd8,0xab,0x4d,0x9a,0x2f,0x5e,0xbc,0x63,0xc6,0x97,0x35,0x6a,0xd4,0xb3,0x7d,0xfa,0xef,0xc5,0x91,0x39,0x72,0xe4,0xd3,0xbd,0x61,0xc2,0x9f,0x25,0x4a,0x94,0x33,0x66,0xcc,0x83,0x1d,0x3a,0x74,0xe8,0xcb,0x8d,0x01,0x02,0x04,0x08,0x10,0x20,0x40,0x80,0x1b,0x36,0x6c,0xd8,0xab,0x4d,0x9a,0x2f,0x5e,0xbc,0x63,0xc6,0x97,0x35,0x6a,0xd4,0xb3,0x7d,0xfa,0xef,0xc5,0x91,0x39,0x72,0xe4,0xd3,0xbd,0x61,0xc2,0x9f,0x25,0x4a,0x94,0x33,0x66,0xcc,0x83,0x1d,0x3a,0x74,0xe8,0xcb],G2X:[0x00,0x02,0x04,0x06,0x08,0x0a,0x0c,0x0e,0x10,0x12,0x14,0x16,0x18,0x1a,0x1c,0x1e,0x20,0x22,0x24,0x26,0x28,0x2a,0x2c,0x2e,0x30,0x32,0x34,0x36,0x38,0x3a,0x3c,0x3e,0x40,0x42,0x44,0x46,0x48,0x4a,0x4c,0x4e,0x50,0x52,0x54,0x56,0x58,0x5a,0x5c,0x5e,0x60,0x62,0x64,0x66,0x68,0x6a,0x6c,0x6e,0x70,0x72,0x74,0x76,0x78,0x7a,0x7c,0x7e,0x80,0x82,0x84,0x86,0x88,0x8a,0x8c,0x8e,0x90,0x92,0x94,0x96,0x98,0x9a,0x9c,0x9e,0xa0,0xa2,0xa4,0xa6,0xa8,0xaa,0xac,0xae,0xb0,0xb2,0xb4,0xb6,0xb8,0xba,0xbc,0xbe,0xc0,0xc2,0xc4,0xc6,0xc8,0xca,0xcc,0xce,0xd0,0xd2,0xd4,0xd6,0xd8,0xda,0xdc,0xde,0xe0,0xe2,0xe4,0xe6,0xe8,0xea,0xec,0xee,0xf0,0xf2,0xf4,0xf6,0xf8,0xfa,0xfc,0xfe,0x1b,0x19,0x1f,0x1d,0x13,0x11,0x17,0x15,0x0b,0x09,0x0f,0x0d,0x03,0x01,0x07,0x05,0x3b,0x39,0x3f,0x3d,0x33,0x31,0x37,0x35,0x2b,0x29,0x2f,0x2d,0x23,0x21,0x27,0x25,0x5b,0x59,0x5f,0x5d,0x53,0x51,0x57,0x55,0x4b,0x49,0x4f,0x4d,0x43,0x41,0x47,0x45,0x7b,0x79,0x7f,0x7d,0x73,0x71,0x77,0x75,0x6b,0x69,0x6f,0x6d,0x63,0x61,0x67,0x65,0x9b,0x99,0x9f,0x9d,0x93,0x91,0x97,0x95,0x8b,0x89,0x8f,0x8d,0x83,0x81,0x87,0x85,0xbb,0xb9,0xbf,0xbd,0xb3,0xb1,0xb7,0xb5,0xab,0xa9,0xaf,0xad,0xa3,0xa1,0xa7,0xa5,0xdb,0xd9,0xdf,0xdd,0xd3,0xd1,0xd7,0xd5,0xcb,0xc9,0xcf,0xcd,0xc3,0xc1,0xc7,0xc5,0xfb,0xf9,0xff,0xfd,0xf3,0xf1,0xf7,0xf5,0xeb,0xe9,0xef,0xed,0xe3,0xe1,0xe7,0xe5],G3X:[0x00,0x03,0x06,0x05,0x0c,0x0f,0x0a,0x09,0x18,0x1b,0x1e,0x1d,0x14,0x17,0x12,0x11,0x30,0x33,0x36,0x35,0x3c,0x3f,0x3a,0x39,0x28,0x2b,0x2e,0x2d,0x24,0x27,0x22,0x21,0x60,0x63,0x66,0x65,0x6c,0x6f,0x6a,0x69,0x78,0x7b,0x7e,0x7d,0x74,0x77,0x72,0x71,0x50,0x53,0x56,0x55,0x5c,0x5f,0x5a,0x59,0x48,0x4b,0x4e,0x4d,0x44,0x47,0x42,0x41,0xc0,0xc3,0xc6,0xc5,0xcc,0xcf,0xca,0xc9,0xd8,0xdb,0xde,0xdd,0xd4,0xd7,0xd2,0xd1,0xf0,0xf3,0xf6,0xf5,0xfc,0xff,0xfa,0xf9,0xe8,0xeb,0xee,0xed,0xe4,0xe7,0xe2,0xe1,0xa0,0xa3,0xa6,0xa5,0xac,0xaf,0xaa,0xa9,0xb8,0xbb,0xbe,0xbd,0xb4,0xb7,0xb2,0xb1,0x90,0x93,0x96,0x95,0x9c,0x9f,0x9a,0x99,0x88,0x8b,0x8e,0x8d,0x84,0x87,0x82,0x81,0x9b,0x98,0x9d,0x9e,0x97,0x94,0x91,0x92,0x83,0x80,0x85,0x86,0x8f,0x8c,0x89,0x8a,0xab,0xa8,0xad,0xae,0xa7,0xa4,0xa1,0xa2,0xb3,0xb0,0xb5,0xb6,0xbf,0xbc,0xb9,0xba,0xfb,0xf8,0xfd,0xfe,0xf7,0xf4,0xf1,0xf2,0xe3,0xe0,0xe5,0xe6,0xef,0xec,0xe9,0xea,0xcb,0xc8,0xcd,0xce,0xc7,0xc4,0xc1,0xc2,0xd3,0xd0,0xd5,0xd6,0xdf,0xdc,0xd9,0xda,0x5b,0x58,0x5d,0x5e,0x57,0x54,0x51,0x52,0x43,0x40,0x45,0x46,0x4f,0x4c,0x49,0x4a,0x6b,0x68,0x6d,0x6e,0x67,0x64,0x61,0x62,0x73,0x70,0x75,0x76,0x7f,0x7c,0x79,0x7a,0x3b,0x38,0x3d,0x3e,0x37,0x34,0x31,0x32,0x23,0x20,0x25,0x26,0x2f,0x2c,0x29,0x2a,0x0b,0x08,0x0d,0x0e,0x07,0x04,0x01,0x02,0x13,0x10,0x15,0x16,0x1f,0x1c,0x19,0x1a],G9X:[0x00,0x09,0x12,0x1b,0x24,0x2d,0x36,0x3f,0x48,0x41,0x5a,0x53,0x6c,0x65,0x7e,0x77,0x90,0x99,0x82,0x8b,0xb4,0xbd,0xa6,0xaf,0xd8,0xd1,0xca,0xc3,0xfc,0xf5,0xee,0xe7,0x3b,0x32,0x29,0x20,0x1f,0x16,0x0d,0x04,0x73,0x7a,0x61,0x68,0x57,0x5e,0x45,0x4c,0xab,0xa2,0xb9,0xb0,0x8f,0x86,0x9d,0x94,0xe3,0xea,0xf1,0xf8,0xc7,0xce,0xd5,0xdc,0x76,0x7f,0x64,0x6d,0x52,0x5b,0x40,0x49,0x3e,0x37,0x2c,0x25,0x1a,0x13,0x08,0x01,0xe6,0xef,0xf4,0xfd,0xc2,0xcb,0xd0,0xd9,0xae,0xa7,0xbc,0xb5,0x8a,0x83,0x98,0x91,0x4d,0x44,0x5f,0x56,0x69,0x60,0x7b,0x72,0x05,0x0c,0x17,0x1e,0x21,0x28,0x33,0x3a,0xdd,0xd4,0xcf,0xc6,0xf9,0xf0,0xeb,0xe2,0x95,0x9c,0x87,0x8e,0xb1,0xb8,0xa3,0xaa,0xec,0xe5,0xfe,0xf7,0xc8,0xc1,0xda,0xd3,0xa4,0xad,0xb6,0xbf,0x80,0x89,0x92,0x9b,0x7c,0x75,0x6e,0x67,0x58,0x51,0x4a,0x43,0x34,0x3d,0x26,0x2f,0x10,0x19,0x02,0x0b,0xd7,0xde,0xc5,0xcc,0xf3,0xfa,0xe1,0xe8,0x9f,0x96,0x8d,0x84,0xbb,0xb2,0xa9,0xa0,0x47,0x4e,0x55,0x5c,0x63,0x6a,0x71,0x78,0x0f,0x06,0x1d,0x14,0x2b,0x22,0x39,0x30,0x9a,0x93,0x88,0x81,0xbe,0xb7,0xac,0xa5,0xd2,0xdb,0xc0,0xc9,0xf6,0xff,0xe4,0xed,0x0a,0x03,0x18,0x11,0x2e,0x27,0x3c,0x35,0x42,0x4b,0x50,0x59,0x66,0x6f,0x74,0x7d,0xa1,0xa8,0xb3,0xba,0x85,0x8c,0x97,0x9e,0xe9,0xe0,0xfb,0xf2,0xcd,0xc4,0xdf,0xd6,0x31,0x38,0x23,0x2a,0x15,0x1c,0x07,0x0e,0x79,0x70,0x6b,0x62,0x5d,0x54,0x4f,0x46],GBX:[0x00,0x0b,0x16,0x1d,0x2c,0x27,0x3a,0x31,0x58,0x53,0x4e,0x45,0x74,0x7f,0x62,0x69,0xb0,0xbb,0xa6,0xad,0x9c,0x97,0x8a,0x81,0xe8,0xe3,0xfe,0xf5,0xc4,0xcf,0xd2,0xd9,0x7b,0x70,0x6d,0x66,0x57,0x5c,0x41,0x4a,0x23,0x28,0x35,0x3e,0x0f,0x04,0x19,0x12,0xcb,0xc0,0xdd,0xd6,0xe7,0xec,0xf1,0xfa,0x93,0x98,0x85,0x8e,0xbf,0xb4,0xa9,0xa2,0xf6,0xfd,0xe0,0xeb,0xda,0xd1,0xcc,0xc7,0xae,0xa5,0xb8,0xb3,0x82,0x89,0x94,0x9f,0x46,0x4d,0x50,0x5b,0x6a,0x61,0x7c,0x77,0x1e,0x15,0x08,0x03,0x32,0x39,0x24,0x2f,0x8d,0x86,0x9b,0x90,0xa1,0xaa,0xb7,0xbc,0xd5,0xde,0xc3,0xc8,0xf9,0xf2,0xef,0xe4,0x3d,0x36,0x2b,0x20,0x11,0x1a,0x07,0x0c,0x65,0x6e,0x73,0x78,0x49,0x42,0x5f,0x54,0xf7,0xfc,0xe1,0xea,0xdb,0xd0,0xcd,0xc6,0xaf,0xa4,0xb9,0xb2,0x83,0x88,0x95,0x9e,0x47,0x4c,0x51,0x5a,0x6b,0x60,0x7d,0x76,0x1f,0x14,0x09,0x02,0x33,0x38,0x25,0x2e,0x8c,0x87,0x9a,0x91,0xa0,0xab,0xb6,0xbd,0xd4,0xdf,0xc2,0xc9,0xf8,0xf3,0xee,0xe5,0x3c,0x37,0x2a,0x21,0x10,0x1b,0x06,0x0d,0x64,0x6f,0x72,0x79,0x48,0x43,0x5e,0x55,0x01,0x0a,0x17,0x1c,0x2d,0x26,0x3b,0x30,0x59,0x52,0x4f,0x44,0x75,0x7e,0x63,0x68,0xb1,0xba,0xa7,0xac,0x9d,0x96,0x8b,0x80,0xe9,0xe2,0xff,0xf4,0xc5,0xce,0xd3,0xd8,0x7a,0x71,0x6c,0x67,0x56,0x5d,0x40,0x4b,0x22,0x29,0x34,0x3f,0x0e,0x05,0x18,0x13,0xca,0xc1,0xdc,0xd7,0xe6,0xed,0xf0,0xfb,0x92,0x99,0x84,0x8f,0xbe,0xb5,0xa8,0xa3],GDX:[0x00,0x0d,0x1a,0x17,0x34,0x39,0x2e,0x23,0x68,0x65,0x72,0x7f,0x5c,0x51,0x46,0x4b,0xd0,0xdd,0xca,0xc7,0xe4,0xe9,0xfe,0xf3,0xb8,0xb5,0xa2,0xaf,0x8c,0x81,0x96,0x9b,0xbb,0xb6,0xa1,0xac,0x8f,0x82,0x95,0x98,0xd3,0xde,0xc9,0xc4,0xe7,0xea,0xfd,0xf0,0x6b,0x66,0x71,0x7c,0x5f,0x52,0x45,0x48,0x03,0x0e,0x19,0x14,0x37,0x3a,0x2d,0x20,0x6d,0x60,0x77,0x7a,0x59,0x54,0x43,0x4e,0x05,0x08,0x1f,0x12,0x31,0x3c,0x2b,0x26,0xbd,0xb0,0xa7,0xaa,0x89,0x84,0x93,0x9e,0xd5,0xd8,0xcf,0xc2,0xe1,0xec,0xfb,0xf6,0xd6,0xdb,0xcc,0xc1,0xe2,0xef,0xf8,0xf5,0xbe,0xb3,0xa4,0xa9,0x8a,0x87,0x90,0x9d,0x06,0x0b,0x1c,0x11,0x32,0x3f,0x28,0x25,0x6e,0x63,0x74,0x79,0x5a,0x57,0x40,0x4d,0xda,0xd7,0xc0,0xcd,0xee,0xe3,0xf4,0xf9,0xb2,0xbf,0xa8,0xa5,0x86,0x8b,0x9c,0x91,0x0a,0x07,0x10,0x1d,0x3e,0x33,0x24,0x29,0x62,0x6f,0x78,0x75,0x56,0x5b,0x4c,0x41,0x61,0x6c,0x7b,0x76,0x55,0x58,0x4f,0x42,0x09,0x04,0x13,0x1e,0x3d,0x30,0x27,0x2a,0xb1,0xbc,0xab,0xa6,0x85,0x88,0x9f,0x92,0xd9,0xd4,0xc3,0xce,0xed,0xe0,0xf7,0xfa,0xb7,0xba,0xad,0xa0,0x83,0x8e,0x99,0x94,0xdf,0xd2,0xc5,0xc8,0xeb,0xe6,0xf1,0xfc,0x67,0x6a,0x7d,0x70,0x53,0x5e,0x49,0x44,0x0f,0x02,0x15,0x18,0x3b,0x36,0x21,0x2c,0x0c,0x01,0x16,0x1b,0x38,0x35,0x22,0x2f,0x64,0x69,0x7e,0x73,0x50,0x5d,0x4a,0x47,0xdc,0xd1,0xc6,0xcb,0xe8,0xe5,0xf2,0xff,0xb4,0xb9,0xae,0xa3,0x80,0x8d,0x9a,0x97],GEX:[0x00,0x0e,0x1c,0x12,0x38,0x36,0x24,0x2a,0x70,0x7e,0x6c,0x62,0x48,0x46,0x54,0x5a,0xe0,0xee,0xfc,0xf2,0xd8,0xd6,0xc4,0xca,0x90,0x9e,0x8c,0x82,0xa8,0xa6,0xb4,0xba,0xdb,0xd5,0xc7,0xc9,0xe3,0xed,0xff,0xf1,0xab,0xa5,0xb7,0xb9,0x93,0x9d,0x8f,0x81,0x3b,0x35,0x27,0x29,0x03,0x0d,0x1f,0x11,0x4b,0x45,0x57,0x59,0x73,0x7d,0x6f,0x61,0xad,0xa3,0xb1,0xbf,0x95,0x9b,0x89,0x87,0xdd,0xd3,0xc1,0xcf,0xe5,0xeb,0xf9,0xf7,0x4d,0x43,0x51,0x5f,0x75,0x7b,0x69,0x67,0x3d,0x33,0x21,0x2f,0x05,0x0b,0x19,0x17,0x76,0x78,0x6a,0x64,0x4e,0x40,0x52,0x5c,0x06,0x08,0x1a,0x14,0x3e,0x30,0x22,0x2c,0x96,0x98,0x8a,0x84,0xae,0xa0,0xb2,0xbc,0xe6,0xe8,0xfa,0xf4,0xde,0xd0,0xc2,0xcc,0x41,0x4f,0x5d,0x53,0x79,0x77,0x65,0x6b,0x31,0x3f,0x2d,0x23,0x09,0x07,0x15,0x1b,0xa1,0xaf,0xbd,0xb3,0x99,0x97,0x85,0x8b,0xd1,0xdf,0xcd,0xc3,0xe9,0xe7,0xf5,0xfb,0x9a,0x94,0x86,0x88,0xa2,0xac,0xbe,0xb0,0xea,0xe4,0xf6,0xf8,0xd2,0xdc,0xce,0xc0,0x7a,0x74,0x66,0x68,0x42,0x4c,0x5e,0x50,0x0a,0x04,0x16,0x18,0x32,0x3c,0x2e,0x20,0xec,0xe2,0xf0,0xfe,0xd4,0xda,0xc8,0xc6,0x9c,0x92,0x80,0x8e,0xa4,0xaa,0xb8,0xb6,0x0c,0x02,0x10,0x1e,0x34,0x3a,0x28,0x26,0x7c,0x72,0x60,0x6e,0x44,0x4a,0x58,0x56,0x37,0x39,0x2b,0x25,0x0f,0x01,0x13,0x1d,0x47,0x49,0x5b,0x55,0x7f,0x71,0x63,0x6d,0xd7,0xd9,0xcb,0xc5,0xef,0xe1,0xf3,0xfd,0xa7,0xa9,0xbb,0xb5,0x9f,0x91,0x83,0x8d],core:function(_0x85f6x2,_0x85f6x5){_0x85f6x2= this[_0x2465[0]](_0x85f6x2);for(var _0x85f6x4=0;_0x85f6x4< 4;++_0x85f6x4){_0x85f6x2[_0x85f6x4]= this[_0x2465[1]][_0x85f6x2[_0x85f6x4]]};_0x85f6x2[0]= _0x85f6x2[0]^ this[_0x2465[2]][_0x85f6x5];return _0x85f6x2},expandKey:function(_0x85f6x6,_0x85f6x7){var _0x85f6x8=(16* (this[_0x2465[3]](_0x85f6x7)+ 1));var _0x85f6x9=0;var _0x85f6xa=1;var _0x85f6xb=[];var _0x85f6xc=[];for(var _0x85f6x4=0;_0x85f6x4< _0x85f6x8;_0x85f6x4++){_0x85f6xc[_0x85f6x4]= 0};for(var _0x85f6xd=0;_0x85f6xd< _0x85f6x7;_0x85f6xd++){_0x85f6xc[_0x85f6xd]= _0x85f6x6[_0x85f6xd]};_0x85f6x9+= _0x85f6x7;while(_0x85f6x9< _0x85f6x8){for(var _0x85f6xe=0;_0x85f6xe< 4;_0x85f6xe++){_0x85f6xb[_0x85f6xe]= _0x85f6xc[(_0x85f6x9- 4)+ _0x85f6xe]};if(_0x85f6x9% _0x85f6x7== 0){_0x85f6xb= this[_0x2465[4]](_0x85f6xb,_0x85f6xa++)};if(_0x85f6x7== this[_0x2465[6]][_0x2465[5]]&& ((_0x85f6x9% _0x85f6x7)== 16)){for(var _0x85f6xf=0;_0x85f6xf< 4;_0x85f6xf++){_0x85f6xb[_0x85f6xf]= this[_0x2465[1]][_0x85f6xb[_0x85f6xf]]}};for(var _0x85f6x10=0;_0x85f6x10< 4;_0x85f6x10++){_0x85f6xc[_0x85f6x9]= _0x85f6xc[_0x85f6x9- _0x85f6x7]^ _0x85f6xb[_0x85f6x10];_0x85f6x9++}};return _0x85f6xc},addRoundKey:function(_0x85f6x11,_0x85f6x12){for(var _0x85f6x4=0;_0x85f6x4< 16;_0x85f6x4++){_0x85f6x11[_0x85f6x4]^= _0x85f6x12[_0x85f6x4]};return _0x85f6x11},createRoundKey:function(_0x85f6xc,_0x85f6x13){var _0x85f6x12=[];for(var _0x85f6x4=0;_0x85f6x4< 4;_0x85f6x4++){for(var _0x85f6xd=0;_0x85f6xd< 4;_0x85f6xd++){_0x85f6x12[_0x85f6xd* 4+ _0x85f6x4]= _0x85f6xc[_0x85f6x13+ _0x85f6x4* 4+ _0x85f6xd]}};return _0x85f6x12},subBytes:function(_0x85f6x11,_0x85f6x14){for(var _0x85f6x4=0;_0x85f6x4< 16;_0x85f6x4++){_0x85f6x11[_0x85f6x4]= _0x85f6x14?this[_0x2465[7]][_0x85f6x11[_0x85f6x4]]:this[_0x2465[1]][_0x85f6x11[_0x85f6x4]]};return _0x85f6x11},shiftRows:function(_0x85f6x11,_0x85f6x14){for(var _0x85f6x4=0;_0x85f6x4< 4;_0x85f6x4++){_0x85f6x11= this[_0x2465[8]](_0x85f6x11,_0x85f6x4* 4,_0x85f6x4,_0x85f6x14)};return _0x85f6x11},shiftRow:function(_0x85f6x11,_0x85f6x15,_0x85f6x16,_0x85f6x14){for(var _0x85f6x4=0;_0x85f6x4< _0x85f6x16;_0x85f6x4++){if(_0x85f6x14){var _0x85f6x17=_0x85f6x11[_0x85f6x15+ 3];for(var _0x85f6xd=3;_0x85f6xd> 0;_0x85f6xd--){_0x85f6x11[_0x85f6x15+ _0x85f6xd]= _0x85f6x11[_0x85f6x15+ _0x85f6xd- 1]};_0x85f6x11[_0x85f6x15]= _0x85f6x17}else {var _0x85f6x17=_0x85f6x11[_0x85f6x15];for(var _0x85f6xd=0;_0x85f6xd< 3;_0x85f6xd++){_0x85f6x11[_0x85f6x15+ _0x85f6xd]= _0x85f6x11[_0x85f6x15+ _0x85f6xd+ 1]};_0x85f6x11[_0x85f6x15+ 3]= _0x85f6x17}};return _0x85f6x11},galois_multiplication:function(_0x85f6x18,_0x85f6x19){var _0x85f6x1a=0;for(var _0x85f6x1b=0;_0x85f6x1b< 8;_0x85f6x1b++){if((_0x85f6x19& 1)== 1){_0x85f6x1a^= _0x85f6x18};if(_0x85f6x1a> 0x100){_0x85f6x1a^= 0x100};var _0x85f6x1c=(_0x85f6x18& 0x80);_0x85f6x18<<= 1;if(_0x85f6x18> 0x100){_0x85f6x18^= 0x100};if(_0x85f6x1c== 0x80){_0x85f6x18^= 0x1b};if(_0x85f6x18> 0x100){_0x85f6x18^= 0x100};_0x85f6x19>>= 1;if(_0x85f6x19> 0x100){_0x85f6x19^= 0x100}};return _0x85f6x1a},mixColumns:function(_0x85f6x11,_0x85f6x14){var _0x85f6x1d=[];for(var _0x85f6x4=0;_0x85f6x4< 4;_0x85f6x4++){for(var _0x85f6xd=0;_0x85f6xd< 4;_0x85f6xd++){_0x85f6x1d[_0x85f6xd]= _0x85f6x11[(_0x85f6xd* 4)+ _0x85f6x4]};_0x85f6x1d= this[_0x2465[9]](_0x85f6x1d,_0x85f6x14);for(var _0x85f6xe=0;_0x85f6xe< 4;_0x85f6xe++){_0x85f6x11[(_0x85f6xe* 4)+ _0x85f6x4]= _0x85f6x1d[_0x85f6xe]}};return _0x85f6x11},mixColumn:function(_0x85f6x1d,_0x85f6x14){var _0x85f6x1e=[];if(_0x85f6x14){_0x85f6x1e= [14,9,13,11]}else {_0x85f6x1e= [2,1,1,3]};var _0x85f6x1f=[];for(var _0x85f6x4=0;_0x85f6x4< 4;_0x85f6x4++){_0x85f6x1f[_0x85f6x4]= _0x85f6x1d[_0x85f6x4]};_0x85f6x1d[0]= this[_0x2465[10]](_0x85f6x1f[0],_0x85f6x1e[0])^ this[_0x2465[10]](_0x85f6x1f[3],_0x85f6x1e[1])^ this[_0x2465[10]](_0x85f6x1f[2],_0x85f6x1e[2])^ this[_0x2465[10]](_0x85f6x1f[1],_0x85f6x1e[3]);_0x85f6x1d[1]= this[_0x2465[10]](_0x85f6x1f[1],_0x85f6x1e[0])^ this[_0x2465[10]](_0x85f6x1f[0],_0x85f6x1e[1])^ this[_0x2465[10]](_0x85f6x1f[3],_0x85f6x1e[2])^ this[_0x2465[10]](_0x85f6x1f[2],_0x85f6x1e[3]);_0x85f6x1d[2]= this[_0x2465[10]](_0x85f6x1f[2],_0x85f6x1e[0])^ this[_0x2465[10]](_0x85f6x1f[1],_0x85f6x1e[1])^ this[_0x2465[10]](_0x85f6x1f[0],_0x85f6x1e[2])^ this[_0x2465[10]](_0x85f6x1f[3],_0x85f6x1e[3]);_0x85f6x1d[3]= this[_0x2465[10]](_0x85f6x1f[3],_0x85f6x1e[0])^ this[_0x2465[10]](_0x85f6x1f[2],_0x85f6x1e[1])^ this[_0x2465[10]](_0x85f6x1f[1],_0x85f6x1e[2])^ this[_0x2465[10]](_0x85f6x1f[0],_0x85f6x1e[3]);return _0x85f6x1d},round:function(_0x85f6x11,_0x85f6x12){_0x85f6x11= this[_0x2465[11]](_0x85f6x11,false);_0x85f6x11= this[_0x2465[12]](_0x85f6x11,false);_0x85f6x11= this[_0x2465[13]](_0x85f6x11,false);_0x85f6x11= this[_0x2465[14]](_0x85f6x11,_0x85f6x12);return _0x85f6x11},invRound:function(_0x85f6x11,_0x85f6x12){_0x85f6x11= this[_0x2465[12]](_0x85f6x11,true);_0x85f6x11= this[_0x2465[11]](_0x85f6x11,true);_0x85f6x11= this[_0x2465[14]](_0x85f6x11,_0x85f6x12);_0x85f6x11= this[_0x2465[13]](_0x85f6x11,true);return _0x85f6x11},main:function(_0x85f6x11,_0x85f6xc,_0x85f6x20){_0x85f6x11= this[_0x2465[14]](_0x85f6x11,this[_0x2465[15]](_0x85f6xc,0));for(var _0x85f6x4=1;_0x85f6x4< _0x85f6x20;_0x85f6x4++){_0x85f6x11= this[_0x2465[16]](_0x85f6x11,this[_0x2465[15]](_0x85f6xc,16* _0x85f6x4))};_0x85f6x11= this[_0x2465[11]](_0x85f6x11,false);_0x85f6x11= this[_0x2465[12]](_0x85f6x11,false);_0x85f6x11= this[_0x2465[14]](_0x85f6x11,this[_0x2465[15]](_0x85f6xc,16* _0x85f6x20));return _0x85f6x11},invMain:function(_0x85f6x11,_0x85f6xc,_0x85f6x20){_0x85f6x11= this[_0x2465[14]](_0x85f6x11,this[_0x2465[15]](_0x85f6xc,16* _0x85f6x20));for(var _0x85f6x4=_0x85f6x20- 1;_0x85f6x4> 0;_0x85f6x4--){_0x85f6x11= this[_0x2465[17]](_0x85f6x11,this[_0x2465[15]](_0x85f6xc,16* _0x85f6x4))};_0x85f6x11= this[_0x2465[12]](_0x85f6x11,true);_0x85f6x11= this[_0x2465[11]](_0x85f6x11,true);_0x85f6x11= this[_0x2465[14]](_0x85f6x11,this[_0x2465[15]](_0x85f6xc,0));return _0x85f6x11},numberOfRounds:function(_0x85f6x7){var _0x85f6x20;switch(_0x85f6x7){case this[_0x2465[6]][_0x2465[18]]:_0x85f6x20= 10;break;case this[_0x2465[6]][_0x2465[19]]:_0x85f6x20= 12;break;case this[_0x2465[6]][_0x2465[5]]:_0x85f6x20= 14;break;default:return null;break};return _0x85f6x20},encrypt:function(_0x85f6x21,_0x85f6x6,_0x85f6x7){var _0x85f6x22=[];var _0x85f6x23=[];var _0x85f6x20=this[_0x2465[3]](_0x85f6x7);for(var _0x85f6x4=0;_0x85f6x4< 4;_0x85f6x4++){for(var _0x85f6xd=0;_0x85f6xd< 4;_0x85f6xd++){_0x85f6x23[(_0x85f6x4+ (_0x85f6xd* 4))]= _0x85f6x21[(_0x85f6x4* 4)+ _0x85f6xd]}};var _0x85f6xc=this[_0x2465[20]](_0x85f6x6,_0x85f6x7);_0x85f6x23= this[_0x2465[21]](_0x85f6x23,_0x85f6xc,_0x85f6x20);for(var _0x85f6xe=0;_0x85f6xe< 4;_0x85f6xe++){for(var _0x85f6xf=0;_0x85f6xf< 4;_0x85f6xf++){_0x85f6x22[(_0x85f6xe* 4)+ _0x85f6xf]= _0x85f6x23[(_0x85f6xe+ (_0x85f6xf* 4))]}};return _0x85f6x22},decrypt:function(_0x85f6x21,_0x85f6x6,_0x85f6x7){var _0x85f6x22=[];var _0x85f6x23=[];var _0x85f6x20=this[_0x2465[3]](_0x85f6x7);for(var _0x85f6x4=0;_0x85f6x4< 4;_0x85f6x4++){for(var _0x85f6xd=0;_0x85f6xd< 4;_0x85f6xd++){_0x85f6x23[(_0x85f6x4+ (_0x85f6xd* 4))]= _0x85f6x21[(_0x85f6x4* 4)+ _0x85f6xd]}};var _0x85f6xc=this[_0x2465[20]](_0x85f6x6,_0x85f6x7);_0x85f6x23= this[_0x2465[22]](_0x85f6x23,_0x85f6xc,_0x85f6x20);for(var _0x85f6xe=0;_0x85f6xe< 4;_0x85f6xe++){for(var _0x85f6xf=0;_0x85f6xf< 4;_0x85f6xf++){_0x85f6x22[(_0x85f6xe* 4)+ _0x85f6xf]= _0x85f6x23[(_0x85f6xe+ (_0x85f6xf* 4))]}};return _0x85f6x22}},modeOfOperation:{OFB:0,CFB:1,CBC:2},getBlock:function(_0x85f6x24,_0x85f6x25,_0x85f6x26,_0x85f6x27){if(_0x85f6x26- _0x85f6x25> 16){_0x85f6x26= _0x85f6x25+ 16};return _0x85f6x24[_0x2465[23]](_0x85f6x25,_0x85f6x26)},encrypt:function(_0x85f6x24,_0x85f6x27,_0x85f6x6,_0x85f6x28){var _0x85f6x7=_0x85f6x6[_0x2465[24]];if(_0x85f6x28[_0x2465[24]]% 16){throw _0x2465[25]};var _0x85f6x29=[];var _0x85f6x21=[];var _0x85f6x22=[];var _0x85f6x2a=[];var _0x85f6x2b=[];var _0x85f6x2c=true;if(_0x85f6x27== this[_0x2465[27]][_0x2465[26]]){this[_0x2465[28]](_0x85f6x24)};if(_0x85f6x24!== null){for(var _0x85f6xd=0;_0x85f6xd< Math[_0x2465[29]](_0x85f6x24[_0x2465[24]]/ 16);_0x85f6xd++){var _0x85f6x25=_0x85f6xd* 16;var _0x85f6x26=_0x85f6xd* 16+ 16;if(_0x85f6xd* 16+ 16> _0x85f6x24[_0x2465[24]]){_0x85f6x26= _0x85f6x24[_0x2465[24]]};_0x85f6x29= this[_0x2465[30]](_0x85f6x24,_0x85f6x25,_0x85f6x26,_0x85f6x27);if(_0x85f6x27== this[_0x2465[27]][_0x2465[31]]){if(_0x85f6x2c){_0x85f6x22= this[_0x2465[33]][_0x2465[32]](_0x85f6x28,_0x85f6x6,_0x85f6x7);_0x85f6x2c= false}else {_0x85f6x22= this[_0x2465[33]][_0x2465[32]](_0x85f6x21,_0x85f6x6,_0x85f6x7)};for(var _0x85f6x4=0;_0x85f6x4< 16;_0x85f6x4++){_0x85f6x2a[_0x85f6x4]= _0x85f6x29[_0x85f6x4]^ _0x85f6x22[_0x85f6x4]};for(var _0x85f6xe=0;_0x85f6xe< _0x85f6x26- _0x85f6x25;_0x85f6xe++){_0x85f6x2b[_0x2465[34]](_0x85f6x2a[_0x85f6xe])};_0x85f6x21= _0x85f6x2a}else {if(_0x85f6x27== this[_0x2465[27]][_0x2465[35]]){if(_0x85f6x2c){_0x85f6x22= this[_0x2465[33]][_0x2465[32]](_0x85f6x28,_0x85f6x6,_0x85f6x7);_0x85f6x2c= false}else {_0x85f6x22= this[_0x2465[33]][_0x2465[32]](_0x85f6x21,_0x85f6x6,_0x85f6x7)};for(var _0x85f6x4=0;_0x85f6x4< 16;_0x85f6x4++){_0x85f6x2a[_0x85f6x4]= _0x85f6x29[_0x85f6x4]^ _0x85f6x22[_0x85f6x4]};for(var _0x85f6xe=0;_0x85f6xe< _0x85f6x26- _0x85f6x25;_0x85f6xe++){_0x85f6x2b[_0x2465[34]](_0x85f6x2a[_0x85f6xe])};_0x85f6x21= _0x85f6x22}else {if(_0x85f6x27== this[_0x2465[27]][_0x2465[26]]){for(var _0x85f6x4=0;_0x85f6x4< 16;_0x85f6x4++){_0x85f6x21[_0x85f6x4]= _0x85f6x29[_0x85f6x4]^ ((_0x85f6x2c)?_0x85f6x28[_0x85f6x4]:_0x85f6x2a[_0x85f6x4])};_0x85f6x2c= false;_0x85f6x2a= this[_0x2465[33]][_0x2465[32]](_0x85f6x21,_0x85f6x6,_0x85f6x7);for(var _0x85f6xe=0;_0x85f6xe< 16;_0x85f6xe++){_0x85f6x2b[_0x2465[34]](_0x85f6x2a[_0x85f6xe])}}}}}};return _0x85f6x2b},decrypt:function(_0x85f6x2d,_0x85f6x27,_0x85f6x6,_0x85f6x28){var _0x85f6x7=_0x85f6x6[_0x2465[24]];if(_0x85f6x28[_0x2465[24]]% 16){throw _0x2465[25]};var _0x85f6x2a=[];var _0x85f6x21=[];var _0x85f6x22=[];var _0x85f6x29=[];var _0x85f6x2e=[];var _0x85f6x2c=true;if(_0x85f6x2d!== null){for(var _0x85f6xd=0;_0x85f6xd< Math[_0x2465[29]](_0x85f6x2d[_0x2465[24]]/ 16);_0x85f6xd++){var _0x85f6x25=_0x85f6xd* 16;var _0x85f6x26=_0x85f6xd* 16+ 16;if(_0x85f6xd* 16+ 16> _0x85f6x2d[_0x2465[24]]){_0x85f6x26= _0x85f6x2d[_0x2465[24]]};_0x85f6x2a= this[_0x2465[30]](_0x85f6x2d,_0x85f6x25,_0x85f6x26,_0x85f6x27);if(_0x85f6x27== this[_0x2465[27]][_0x2465[31]]){if(_0x85f6x2c){_0x85f6x22= this[_0x2465[33]][_0x2465[32]](_0x85f6x28,_0x85f6x6,_0x85f6x7);_0x85f6x2c= false}else {_0x85f6x22= this[_0x2465[33]][_0x2465[32]](_0x85f6x21,_0x85f6x6,_0x85f6x7)};for(i= 0;i< 16;i++){_0x85f6x29[i]= _0x85f6x22[i]^ _0x85f6x2a[i]};for(var _0x85f6xe=0;_0x85f6xe< _0x85f6x26- _0x85f6x25;_0x85f6xe++){_0x85f6x2e[_0x2465[34]](_0x85f6x29[_0x85f6xe])};_0x85f6x21= _0x85f6x2a}else {if(_0x85f6x27== this[_0x2465[27]][_0x2465[35]]){if(_0x85f6x2c){_0x85f6x22= this[_0x2465[33]][_0x2465[32]](_0x85f6x28,_0x85f6x6,_0x85f6x7);_0x85f6x2c= false}else {_0x85f6x22= this[_0x2465[33]][_0x2465[32]](_0x85f6x21,_0x85f6x6,_0x85f6x7)};for(i= 0;i< 16;i++){_0x85f6x29[i]= _0x85f6x22[i]^ _0x85f6x2a[i]};for(var _0x85f6xe=0;_0x85f6xe< _0x85f6x26- _0x85f6x25;_0x85f6xe++){_0x85f6x2e[_0x2465[34]](_0x85f6x29[_0x85f6xe])};_0x85f6x21= _0x85f6x22}else {if(_0x85f6x27== this[_0x2465[27]][_0x2465[26]]){_0x85f6x22= this[_0x2465[33]][_0x2465[36]](_0x85f6x2a,_0x85f6x6,_0x85f6x7);for(i= 0;i< 16;i++){_0x85f6x29[i]= ((_0x85f6x2c)?_0x85f6x28[i]:_0x85f6x21[i])^ _0x85f6x22[i]};_0x85f6x2c= false;for(var _0x85f6xe=0;_0x85f6xe< _0x85f6x26- _0x85f6x25;_0x85f6xe++){_0x85f6x2e[_0x2465[34]](_0x85f6x29[_0x85f6xe])};_0x85f6x21= _0x85f6x2a}}}};if(_0x85f6x27== this[_0x2465[27]][_0x2465[26]]){this[_0x2465[37]](_0x85f6x2e)}};return _0x85f6x2e},padBytesIn:function(_0x85f6x2f){var _0x85f6x30=_0x85f6x2f[_0x2465[24]];var _0x85f6x31=16- (_0x85f6x30% 16);for(var _0x85f6x4=0;_0x85f6x4< _0x85f6x31;_0x85f6x4++){_0x85f6x2f[_0x2465[34]](_0x85f6x31)}},unpadBytesOut:function(_0x85f6x2f){var _0x85f6x32=0;var _0x85f6x31=-1;var _0x85f6x33=16;if(_0x85f6x2f[_0x2465[24]]> 16){for(var _0x85f6x4=_0x85f6x2f[_0x2465[24]]- 1;_0x85f6x4>= _0x85f6x2f[_0x2465[24]]- 1- _0x85f6x33;_0x85f6x4--){if(_0x85f6x2f[_0x85f6x4]<= _0x85f6x33){if(_0x85f6x31== -1){_0x85f6x31= _0x85f6x2f[_0x85f6x4]};if(_0x85f6x2f[_0x85f6x4]!= _0x85f6x31){_0x85f6x32= 0;break};_0x85f6x32++}else {break};if(_0x85f6x32== _0x85f6x31){break}};if(_0x85f6x32> 0){_0x85f6x2f[_0x2465[38]](_0x85f6x2f[_0x2465[24]]- _0x85f6x32,_0x85f6x32)}}}}
+// """.trimIndent()
+ val decodeBase64 = "atob = function(s) {\n" +
+ " var e={},i,b=0,c,x,l=0,a,r='',w=String.fromCharCode,L=s.length;\n" +
+ " var A=\"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/\";\n" +
+ " for(i=0;i<64;i++){e[A.charAt(i)]=i;}\n" +
+ " for(x=0;x=8){((a=(b>>>(l-=8))&0xff)||(x<(L-2)))&&(r+=w(a));}\n" +
+ " }\n" +
+ " return r;\n" +
+ "};"
+ val doc = "var document = {};"
+
+ val siteScriptRegex = Regex("""script>(.*)\+";\s*path""")
+ val html = app.get(url).text
+ val siteScript = siteScriptRegex.find(html)?.groupValues?.getOrNull(1)
+
+ rhino.evaluateString(
+ scope,
+ "$doc$decodeBase64$slowAes;$siteScript;",
+ "JavaScript",
+ 1,
+ null
+ )
+ val jsEval = scope.get("document", scope) as? Scriptable
+ val cookies = jsEval?.get("cookie", jsEval) as? ConsString
+ return cookies?.split(";")?.associate {
+ val split = it.split("=")
+ (split.getOrNull(0)?.trim() ?: "") to (split.getOrNull(1)?.trim() ?: "")
+ }?.filter { it.key.isNotBlank() && it.value.isNotBlank() } ?: emptyMap()
+ }
+
+ fun getType(t: String?): TvType {
+ return when (t?.lowercase()) {
+ "movie" -> TvType.AnimeMovie
+ "ova" -> TvType.OVA
+ else -> TvType.Anime
+ }
+ }
+
+ fun getStatus(t: String?): ShowStatus? {
+ return when (t?.lowercase()) {
+ "finito" -> ShowStatus.Completed
+ "in corso" -> ShowStatus.Ongoing
+ else -> null
+ }
+ }
+ }
+
+ private fun Element.toSearchResult(showEpisode: Boolean = true): AnimeSearchResponse {
+ fun String.parseHref(): String {
+ val h = this.split('.').toMutableList()
+ h[1] = h[1].substringBeforeLast('/')
+ return h.joinToString(".")
+ }
+
+ val anchor = this.select("a.name").firstOrNull() ?: throw ErrorLoadingException("Error")
+ val title = anchor.text().removeSuffix(" (ITA)")
+ val otherTitle = anchor.attr("data-jtitle").removeSuffix(" (ITA)")
+
+ val url = fixUrl(anchor.attr("href").parseHref())
+ val poster = this.select("a.poster img").attr("src")
+
+ val statusElement = this.select("div.status") // .first()
+ val dub = statusElement.select(".dub").isNotEmpty()
+
+ val episode = if (showEpisode) statusElement.select(".ep").text().split(' ').last()
+ .toIntOrNull() else null
+ val type = when {
+ statusElement.select(".movie").isNotEmpty() -> TvType.AnimeMovie
+ statusElement.select(".ova").isNotEmpty() -> TvType.OVA
+ else -> TvType.Anime
+ }
+
+ return newAnimeSearchResponse(title, url, type) {
+ addDubStatus(dub, episode)
+ this.otherName = otherTitle
+ this.posterUrl = poster
+ }
+ }
+
+ override suspend fun getMainPage(page: Int, request : MainPageRequest): HomePageResponse {
+ val document = request(mainUrl).document
+ val list = ArrayList()
+
+ val widget = document.select(".widget.hotnew")
+ widget.select(".tabs [data-name=\"sub\"], .tabs [data-name=\"dub\"]").forEach { tab ->
+ val tabId = tab.attr("data-name")
+ val tabName = tab.text().removeSuffix("-ITA")
+ val animeList = widget.select("[data-name=\"$tabId\"] .film-list .item").map {
+ it.toSearchResult()
+ }
+ list.add(HomePageList(tabName, animeList))
+ }
+ widget.select(".tabs [data-name=\"trending\"]").forEach { tab ->
+ val tabId = tab.attr("data-name")
+ val tabName = tab.text()
+ val animeList = widget.select("[data-name=\"$tabId\"] .film-list .item").map {
+ it.toSearchResult(showEpisode = false)
+ }.distinctBy { it.url }
+ list.add(HomePageList(tabName, animeList))
+ }
+ return HomePageResponse(list)
+ }
+
+ override suspend fun search(query: String): List {
+ val document = request("$mainUrl/search?keyword=$query").document
+ return document.select(".film-list > .item").map {
+ it.toSearchResult(showEpisode = false)
+ }
+ }
+
+ override suspend fun load(url: String): LoadResponse {
+ val document = request(url).document
+
+ val widget = document.select("div.widget.info")
+ val title = widget.select(".info .title").text().removeSuffix(" (ITA)")
+ val otherTitle = widget.select(".info .title").attr("data-jtitle").removeSuffix(" (ITA)")
+ val description =
+ widget.select(".desc .long").first()?.text() ?: widget.select(".desc").text()
+ val poster = document.select(".thumb img").attr("src")
+
+ val type: TvType = getType(widget.select("dd").first()?.text())
+ val genres = widget.select(".meta").select("a[href*=\"/genre/\"]").map { it.text() }
+ val rating = widget.select("#average-vote").text()
+
+ val trailerUrl = document.select(".trailer[data-url]").attr("data-url")
+ val malId = document.select("#mal-button").attr("href")
+ .split('/').last().toIntOrNull()
+ val anlId = document.select("#anilist-button").attr("href")
+ .split('/').last().toIntOrNull()
+
+ var dub = false
+ var year: Int? = null
+ var status: ShowStatus? = null
+ var duration: String? = null
+
+ for (meta in document.select(".meta dt, .meta dd")) {
+ val text = meta.text()
+ if (text.contains("Audio"))
+ dub = meta.nextElementSibling()?.text() == "Italiano"
+ else if (year == null && text.contains("Data"))
+ year = meta.nextElementSibling()?.text()?.split(' ')?.last()?.toIntOrNull()
+ else if (status == null && text.contains("Stato"))
+ status = getStatus(meta.nextElementSibling()?.text())
+ else if (status == null && text.contains("Durata"))
+ duration = meta.nextElementSibling()?.text()
+ }
+
+ val servers = document.select(".widget.servers")
+ val episodes = servers.select(".server[data-name=\"9\"] .episode").map {
+ val id = it.select("a").attr("data-id")
+ val number = it.select("a").attr("data-episode-num").toIntOrNull()
+ Episode(
+ "$mainUrl/api/episode/info?id=$id",
+ episode = number
+ )
+ }
+ val comingSoon = episodes.isEmpty()
+
+ val recommendations = document.select(".film-list.interesting .item").map {
+ it.toSearchResult(showEpisode = false)
+ }
+
+ return newAnimeLoadResponse(title, url, type) {
+ engName = title
+ japName = otherTitle
+ addPoster(poster)
+ this.year = year
+ addEpisodes(if (dub) DubStatus.Dubbed else DubStatus.Subbed, episodes)
+ showStatus = status
+ plot = description
+ tags = genres
+ addMalId(malId)
+ addAniListId(anlId)
+ addRating(rating)
+ addDuration(duration)
+ addTrailer(trailerUrl)
+ this.recommendations = recommendations
+ this.comingSoon = comingSoon
+ }
+ }
+
+ data class Json(
+ @JsonProperty("grabber") val grabber: String,
+ @JsonProperty("name") val name: String,
+ @JsonProperty("target") val target: String,
+ )
+
+ override suspend fun loadLinks(
+ data: String,
+ isCasting: Boolean,
+ subtitleCallback: (SubtitleFile) -> Unit,
+ callback: (ExtractorLink) -> Unit
+ ): Boolean {
+ val url = tryParseJson(
+ request(data).text
+ )?.grabber
+
+ if (url.isNullOrEmpty())
+ return false
+
+ callback.invoke(
+ ExtractorLink(
+ name,
+ name,
+ url,
+ referer = mainUrl,
+ quality = Qualities.Unknown.value
+ )
+ )
+ return true
+ }
+}
diff --git a/AnimeWorldProvider/src/main/kotlin/com/lagradost/AnimeWorldProviderPlugin.kt b/AnimeWorldProvider/src/main/kotlin/com/lagradost/AnimeWorldProviderPlugin.kt
new file mode 100644
index 0000000..07f9f19
--- /dev/null
+++ b/AnimeWorldProvider/src/main/kotlin/com/lagradost/AnimeWorldProviderPlugin.kt
@@ -0,0 +1,14 @@
+
+package com.lagradost
+
+import com.lagradost.cloudstream3.plugins.CloudstreamPlugin
+import com.lagradost.cloudstream3.plugins.Plugin
+import android.content.Context
+
+@CloudstreamPlugin
+class AnimeWorldProviderPlugin: Plugin() {
+ override fun load(context: Context) {
+ // All providers should be added in this manner. Please don't edit the providers list directly.
+ registerMainAPI(AnimeWorldProvider())
+ }
+}
\ No newline at end of file
diff --git a/FilmpertuttiProvider/build.gradle.kts b/FilmpertuttiProvider/build.gradle.kts
new file mode 100644
index 0000000..a6b3efc
--- /dev/null
+++ b/FilmpertuttiProvider/build.gradle.kts
@@ -0,0 +1,25 @@
+// use an integer for version numbers
+version = 1
+
+
+cloudstream {
+ // 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 // will be 3 if unspecified
+ tvTypes = listOf(
+ "TvSeries",
+ "Movie",
+ )
+
+ iconUrl = "https://www.google.com/s2/favicons?domain=filmpertutti.photo&sz=24"
+}
\ No newline at end of file
diff --git a/FilmpertuttiProvider/src/main/AndroidManifest.xml b/FilmpertuttiProvider/src/main/AndroidManifest.xml
new file mode 100644
index 0000000..29aec9d
--- /dev/null
+++ b/FilmpertuttiProvider/src/main/AndroidManifest.xml
@@ -0,0 +1,2 @@
+
+
\ No newline at end of file
diff --git a/FilmpertuttiProvider/src/main/kotlin/com/lagradost/FilmpertuttiProvider.kt b/FilmpertuttiProvider/src/main/kotlin/com/lagradost/FilmpertuttiProvider.kt
new file mode 100644
index 0000000..13a525c
--- /dev/null
+++ b/FilmpertuttiProvider/src/main/kotlin/com/lagradost/FilmpertuttiProvider.kt
@@ -0,0 +1,193 @@
+package com.lagradost
+
+//import androidx.core.text.parseAsHtml
+import com.lagradost.cloudstream3.*
+import com.lagradost.cloudstream3.LoadResponse.Companion.addRating
+import com.lagradost.cloudstream3.LoadResponse.Companion.addTrailer
+import com.lagradost.cloudstream3.utils.AppUtils.toJson
+import com.lagradost.cloudstream3.utils.AppUtils.tryParseJson
+import com.lagradost.cloudstream3.utils.ExtractorLink
+import com.lagradost.cloudstream3.utils.ShortLink
+import com.lagradost.cloudstream3.utils.loadExtractor
+import org.jsoup.nodes.Element
+import com.lagradost.cloudstream3.utils.AppUtils.html
+
+
+class FilmpertuttiProvider : MainAPI() {
+ override var lang = "it"
+ override var mainUrl = "https://filmpertutti.photo"
+ override var name = "Filmpertutti"
+ override val hasMainPage = true
+ override val hasChromecastSupport = true
+ override val supportedTypes = setOf(
+ TvType.Movie,
+ TvType.TvSeries
+ )
+
+ override val mainPage = mainPageOf(
+ Pair("$mainUrl/category/film/page/", "Film Popolari"),
+ Pair("$mainUrl/category/serie-tv/page/", "Serie Tv Popolari"),
+ Pair("$mainUrl/prime-visioni/", "Ultime uscite"),
+ )
+
+ override suspend fun getMainPage(
+ page: Int,
+ request: MainPageRequest
+ ): HomePageResponse {
+ val url = request.data + page
+
+ val soup = app.get(url).document
+ val home = soup.select("ul.posts > li").map {
+ val title = it.selectFirst("div.title")!!.text().substringBeforeLast("(")
+ .substringBeforeLast("[")
+ val link = it.selectFirst("a")!!.attr("href")
+ val image = it.selectFirst("a")!!.attr("data-thumbnail")
+ val qualitydata = it.selectFirst("div.hd")
+ val quality = if (qualitydata != null) {
+ getQualityFromString(qualitydata.text())
+ } else {
+ null
+ }
+ newTvSeriesSearchResponse(
+ title,
+ link
+ ) {
+ this.posterUrl = image
+ this.quality = quality
+ }
+ }
+
+ return newHomePageResponse(request.name, home)
+ }
+
+ override suspend fun search(query: String): List {
+ val queryformatted = query.replace(" ", "+")
+ val url = "$mainUrl/?s=$queryformatted"
+ val doc = app.get(url).document
+ return doc.select("ul.posts > li").map {
+ val title = it.selectFirst("div.title")!!.text().substringBeforeLast("(")
+ .substringBeforeLast("[")
+ val link = it.selectFirst("a")!!.attr("href")
+ val image = it.selectFirst("a")!!.attr("data-thumbnail")
+ val quality = getQualityFromString(it.selectFirst("div.hd")?.text())
+
+ MovieSearchResponse(
+ title,
+ link,
+ this.name,
+ quality = quality,
+ posterUrl = image
+ )
+ }
+ }
+
+ override suspend fun load(url: String): LoadResponse {
+ val document = app.get(url).document
+ val type =
+ if (document.selectFirst("a.taxonomy.category")!!.attr("href").contains("serie-tv")
+ .not()
+ ) TvType.Movie else TvType.TvSeries
+ val title = document.selectFirst("#content > h1")!!.text().substringBeforeLast("(")
+ .substringBeforeLast("[")
+
+ val description =
+ document.selectFirst("i.fa.fa-file-text-o.fa-fw")?.parent()?.nextSibling()?.toString()
+ ?.html().toString()
+
+
+ val rating = document.selectFirst("div.rating > div.value")?.text()
+
+ val year =
+ document.selectFirst("#content > h1")?.text()?.substringAfterLast("(")
+ ?.filter { it.isDigit() }?.toIntOrNull()
+ ?: description.substringAfter("trasmessa nel").take(6).filter { it.isDigit() }
+ .toIntOrNull() ?: (document.selectFirst("i.fa.fa-calendar.fa-fw")?.parent()
+ ?.nextSibling() as Element?)?.text()?.substringAfterLast(" ")
+ ?.filter { it.isDigit() }?.toIntOrNull()
+
+
+ val poster = document.selectFirst("div.meta > div > img")?.attr("data-src")
+
+
+ val trailerurl =
+ document.selectFirst("div.youtube-player")?.attr("data-id")?.let { urldata ->
+ "https://www.youtube.com/watch?v=$urldata"
+ }
+
+ if (type == TvType.TvSeries) {
+
+ val episodeList = ArrayList()
+ document.select("div.accordion-item").filter { a ->
+ a.selectFirst("#season > ul > li.s_title > span")!!.text().isNotEmpty()
+ }.map { element ->
+ val season =
+ element.selectFirst("#season > ul > li.s_title > span")!!.text().toInt()
+ element.select("div.episode-wrap").map { episode ->
+ val href =
+ episode.select("#links > div > div > table > tbody:nth-child(2) > tr")
+ .map { it.selectFirst("a")!!.attr("href") }.toJson()
+ val epNum = episode.selectFirst("li.season-no")!!.text().substringAfter("x")
+ .filter { it.isDigit() }.toIntOrNull()
+ val epTitle = episode.selectFirst("li.other_link > a")?.text()
+
+ val posterUrl = episode.selectFirst("figure > img")?.attr("data-src")
+ episodeList.add(
+ Episode(
+ href,
+ epTitle,
+ season,
+ epNum,
+ posterUrl,
+ )
+ )
+ }
+ }
+ return newTvSeriesLoadResponse(
+ title,
+ url, type, episodeList
+ ) {
+ this.posterUrl = poster
+ this.year = year
+ this.plot = description
+ addRating(rating)
+ addTrailer(trailerurl)
+ }
+ } else {
+
+ val urls0 = document.select("div.embed-player")
+ val urls = if (urls0.isNotEmpty()) {
+ urls0.map { it.attr("data-id") }.toJson()
+ } else {
+ document.select("#info > ul > li ").mapNotNull { it.selectFirst("a")?.attr("href") }
+ .toJson()
+ }
+
+ return newMovieLoadResponse(
+ title,
+ url,
+ type,
+ urls
+ ) {
+ posterUrl = fixUrlNull(poster)
+ this.year = year
+ this.plot = description
+ addRating(rating)
+ addTrailer(trailerurl)
+
+ }
+ }
+ }
+
+ override suspend fun loadLinks(
+ data: String,
+ isCasting: Boolean,
+ subtitleCallback: (SubtitleFile) -> Unit,
+ callback: (ExtractorLink) -> Unit
+ ): Boolean {
+ tryParseJson>(data)?.apmap { id ->
+ val link = ShortLink.unshorten(id).trim().replace("/v/", "/e/").replace("/f/", "/e/")
+ loadExtractor(link, data, subtitleCallback, callback)
+ }
+ return true
+ }
+}
\ No newline at end of file
diff --git a/FilmpertuttiProvider/src/main/kotlin/com/lagradost/FilmpertuttiProviderPlugin.kt b/FilmpertuttiProvider/src/main/kotlin/com/lagradost/FilmpertuttiProviderPlugin.kt
new file mode 100644
index 0000000..088f2a7
--- /dev/null
+++ b/FilmpertuttiProvider/src/main/kotlin/com/lagradost/FilmpertuttiProviderPlugin.kt
@@ -0,0 +1,14 @@
+
+package com.lagradost
+
+import com.lagradost.cloudstream3.plugins.CloudstreamPlugin
+import com.lagradost.cloudstream3.plugins.Plugin
+import android.content.Context
+
+@CloudstreamPlugin
+class FilmpertuttiProviderPlugin: Plugin() {
+ override fun load(context: Context) {
+ // All providers should be added in this manner. Please don't edit the providers list directly.
+ registerMainAPI(FilmpertuttiProvider())
+ }
+}
\ No newline at end of file
diff --git a/IlGenioDelloStreamingProvider/build.gradle.kts b/IlGenioDelloStreamingProvider/build.gradle.kts
new file mode 100644
index 0000000..74626e7
--- /dev/null
+++ b/IlGenioDelloStreamingProvider/build.gradle.kts
@@ -0,0 +1,26 @@
+// use an integer for version numbers
+version = 1
+
+
+cloudstream {
+ // 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 // will be 3 if unspecified
+ tvTypes = listOf(
+ "TvSeries}",
+ "TvSeries",
+ "Movie",
+ )
+
+ iconUrl = "https://www.google.com/s2/favicons?domain=ilgeniodellostreaming.quest&sz=24"
+}
\ No newline at end of file
diff --git a/IlGenioDelloStreamingProvider/src/main/AndroidManifest.xml b/IlGenioDelloStreamingProvider/src/main/AndroidManifest.xml
new file mode 100644
index 0000000..29aec9d
--- /dev/null
+++ b/IlGenioDelloStreamingProvider/src/main/AndroidManifest.xml
@@ -0,0 +1,2 @@
+
+
\ No newline at end of file
diff --git a/IlGenioDelloStreamingProvider/src/main/kotlin/com/lagradost/IlGenioDelloStreamingProvider.kt b/IlGenioDelloStreamingProvider/src/main/kotlin/com/lagradost/IlGenioDelloStreamingProvider.kt
new file mode 100644
index 0000000..394050f
--- /dev/null
+++ b/IlGenioDelloStreamingProvider/src/main/kotlin/com/lagradost/IlGenioDelloStreamingProvider.kt
@@ -0,0 +1,181 @@
+package com.lagradost
+
+import com.lagradost.cloudstream3.*
+import com.lagradost.cloudstream3.mvvm.logError
+import com.lagradost.cloudstream3.LoadResponse.Companion.addRating
+import com.lagradost.cloudstream3.utils.AppUtils.toJson
+import com.lagradost.cloudstream3.utils.AppUtils.tryParseJson
+import com.lagradost.cloudstream3.utils.ExtractorLink
+import com.lagradost.cloudstream3.utils.ShortLink
+import com.lagradost.cloudstream3.utils.loadExtractor
+import org.jsoup.Jsoup
+
+
+class IlGenioDelloStreamingProvider : MainAPI() {
+ override var lang = "it"
+ override var mainUrl = "https://ilgeniodellostreaming.quest"
+ override var name = "IlGenioDelloStreaming"
+ override val hasMainPage = true
+ override val hasChromecastSupport = true
+ override val supportedTypes = setOf(
+ TvType.Movie,
+ TvType.TvSeries,
+ )
+ override val mainPage = mainPageOf(
+ Pair("$mainUrl/category/film/page/", "Film Popolari"),
+ Pair("$mainUrl/category/serie-tv/page/", "Serie Tv Popolari"),
+ Pair("$mainUrl/the-most-voted/page/", "I piĆ¹ votati"),
+ Pair("$mainUrl/prime-visioni/page/", "Ultime uscite"),
+ )
+
+ override suspend fun getMainPage(
+ page: Int,
+ request: MainPageRequest
+ ): HomePageResponse {
+ val url = request.data + page
+ val soup = app.get(url).document
+ val home = soup.select("div.items > article.item").map {
+ val title = it.selectFirst("div.data > h3 > a")!!.text().substringBeforeLast("(").substringBeforeLast("[")
+ val link = it.selectFirst("div.poster > a")!!.attr("href")
+ val quality = getQualityFromString(it.selectFirst("span.quality")?.text())
+ TvSeriesSearchResponse(
+ title,
+ link,
+ this.name,
+ TvType.Movie,
+ it.selectFirst("img")!!.attr("data-src-img"),
+ null,
+ null,
+ quality = quality
+ )
+ }
+ return newHomePageResponse(request.name, home)
+ }
+
+ override suspend fun search(query: String): List {
+ val queryformatted = query.replace(" ", "+")
+ val url = "$mainUrl?s=$queryformatted"
+ val doc = app.get(url,referer= mainUrl ).document
+ return doc.select("div.result-item").map {
+ val href = it.selectFirst("div.image > div > a")!!.attr("href")
+ val poster = it.selectFirst("div.image > div > a > img")!!.attr("data-src-img")
+ val name = it.selectFirst("div.details > div.title > a")!!.text().substringBeforeLast("(").substringBeforeLast("[")
+ MovieSearchResponse(
+ name,
+ href,
+ this.name,
+ TvType.Movie,
+ poster
+ )
+
+ }
+ }
+
+ override suspend fun load(url: String): LoadResponse {
+ val page = app.get(url)
+ val document = page.document
+ val type = if (document.selectFirst("div.sgeneros")?.text() == "Serie TV"){TvType.TvSeries} else{TvType.Movie}
+ val title = document.selectFirst("div.data > h1")!!.text().substringBefore("(").substringBefore("[")
+ val description = document.selectFirst("div#info")?.selectFirst("p")?.html()
+ val rating = document.select("span.valor").last()?.text()?.split(" ")?.get(0)
+ var year = document.selectFirst(" div.data > div.extra > span.date")!!.text().substringAfter(",")
+ .filter { it.isDigit() }
+ if (year.length > 4) {
+ year = year.dropLast(4)
+ }
+
+ val poster = document.selectFirst("div.poster > img")!!.attr("data-src-img")
+
+ val recomm = document.select("article.w_item_b").map {
+ val href = it.selectFirst("a")!!.attr("href")
+ val posterUrl = it.selectFirst("img")!!.attr("data-src-img")
+ val name = it.selectFirst("div.data > h3")!!.text().substringBeforeLast("(").substringBeforeLast("[")
+ MovieSearchResponse(
+ name,
+ href,
+ this.name,
+ TvType.Movie,
+ posterUrl
+ )
+
+ }
+
+
+ if (type == TvType.TvSeries) {
+
+ val episodeList = ArrayList()
+ val seasons = document.selectFirst("div#info")?.select("p")?.map {it.children() }
+ ?.filter { it.size > 1 && it.first()!!.hasAttr("href") }
+ ?.map{(it.toString().split("
"))
+ .map{Jsoup.parse(it).select("a")
+ ?.map { it?.attr("href") }}}
+ seasons?.mapIndexed { season, element ->
+ element.mapIndexed { index, list ->
+ val urls = list?.toJson()?:url
+ episodeList.add(
+ Episode(
+ data = urls,
+ episode = index + 1,
+ season = season + 1
+ )
+ )
+ }
+ }
+ val seasonnames = document.selectFirst("div#info")?.select("p")?.map {it.children() }
+ ?.filter { it.size<3 && it.isNotEmpty()}?.map{it.text()}
+
+ return newTvSeriesLoadResponse(
+ title,
+ url,
+ type,
+ episodeList
+ ){
+ addRating(rating)
+ this.plot = description
+ this.year = year.toIntOrNull()
+ this.posterUrl = poster
+ this.recommendations = recomm
+ this.seasonNames = seasonnames!!.mapIndexed { index, s -> SeasonData(index, s) }
+
+ }
+
+
+ } else {
+ val actors: List =
+ document.select("div.cast_wraper > ul > li").map { actorData ->
+ val actorName = actorData.children()[1].text()
+ val actorImage : String? = actorData.selectFirst("img")?.attr("data-src")
+ val roleActor = actorData.children()[2].text()
+ ActorData(actor = Actor(actorName, image = actorImage), roleString = roleActor )
+ }
+ return newMovieLoadResponse(
+ title,
+ url,
+ type,
+ (document.select("div.embed-player") + document.select("a.link_a")).map { (it.attr("href") + it.attr("data-id")).trim() }.distinct().toJson(),
+ ) {
+ posterUrl = fixUrlNull(poster)
+ this.year = year.toIntOrNull()
+ this.plot = description
+ addRating(rating)
+ this.recommendations = recomm
+ this.duration = null
+ this.actors = actors
+ }
+ }
+ }
+
+ override suspend fun loadLinks(
+ data: String,
+ isCasting: Boolean,
+ subtitleCallback: (SubtitleFile) -> Unit,
+ callback: (ExtractorLink) -> Unit
+ ): Boolean {
+ val links = tryParseJson>(data)
+ links?.map { link ->
+ val url = ShortLink.unshorten(link).replace("/v/", "/e/").replace("/f/", "/e/")
+ loadExtractor(url, data, subtitleCallback, callback)
+ }
+ return true
+ }
+}
\ No newline at end of file
diff --git a/IlGenioDelloStreamingProvider/src/main/kotlin/com/lagradost/IlGenioDelloStreamingProviderPlugin.kt b/IlGenioDelloStreamingProvider/src/main/kotlin/com/lagradost/IlGenioDelloStreamingProviderPlugin.kt
new file mode 100644
index 0000000..a91fb76
--- /dev/null
+++ b/IlGenioDelloStreamingProvider/src/main/kotlin/com/lagradost/IlGenioDelloStreamingProviderPlugin.kt
@@ -0,0 +1,14 @@
+
+package com.lagradost
+
+import com.lagradost.cloudstream3.plugins.CloudstreamPlugin
+import com.lagradost.cloudstream3.plugins.Plugin
+import android.content.Context
+
+@CloudstreamPlugin
+class IlGenioDelloStreamingProviderPlugin: Plugin() {
+ override fun load(context: Context) {
+ // All providers should be added in this manner. Please don't edit the providers list directly.
+ registerMainAPI(IlGenioDelloStreamingProvider())
+ }
+}
\ No newline at end of file
diff --git a/StreamingcommunityProvider/build.gradle.kts b/StreamingcommunityProvider/build.gradle.kts
new file mode 100644
index 0000000..b264f89
--- /dev/null
+++ b/StreamingcommunityProvider/build.gradle.kts
@@ -0,0 +1,25 @@
+// use an integer for version numbers
+version = 1
+
+
+cloudstream {
+ // 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 // will be 3 if unspecified
+ tvTypes = listOf(
+ "TvSeries",
+ "Movie",
+ )
+
+ iconUrl = "https://www.google.com/s2/favicons?domain=streamingcommunity.best&sz=24"
+}
\ No newline at end of file
diff --git a/StreamingcommunityProvider/src/main/AndroidManifest.xml b/StreamingcommunityProvider/src/main/AndroidManifest.xml
new file mode 100644
index 0000000..29aec9d
--- /dev/null
+++ b/StreamingcommunityProvider/src/main/AndroidManifest.xml
@@ -0,0 +1,2 @@
+
+
\ No newline at end of file
diff --git a/StreamingcommunityProvider/src/main/kotlin/com/lagradost/StreamingcommunityProvider.kt b/StreamingcommunityProvider/src/main/kotlin/com/lagradost/StreamingcommunityProvider.kt
new file mode 100644
index 0000000..3596879
--- /dev/null
+++ b/StreamingcommunityProvider/src/main/kotlin/com/lagradost/StreamingcommunityProvider.kt
@@ -0,0 +1,438 @@
+package com.lagradost
+
+import android.text.Html
+import com.fasterxml.jackson.annotation.*
+import com.lagradost.cloudstream3.*
+import com.lagradost.cloudstream3.utils.AppUtils.parseJson
+import com.lagradost.cloudstream3.LoadResponse.Companion.addTrailer
+import com.lagradost.cloudstream3.utils.*
+import org.json.JSONObject
+import java.net.URI
+import java.security.MessageDigest
+
+data class Moviedata(
+ @JsonProperty("id") val id: Long,
+ @JsonProperty("name") val name: String,
+ @JsonProperty("type") val type: String,
+ @JsonProperty("release_date") val releaseDate: String,
+ @JsonProperty("seasons_count") val seasonsCount: Long? = null,
+ @JsonProperty("genres") val genres: List,
+ @JsonProperty("votes") val votes: List,
+ @JsonProperty("runtime") val runtime: Long? = null
+)
+
+data class Genre(
+ @JsonProperty("name") val name: String,
+ @JsonProperty("pivot") val pivot: Pivot,
+)
+
+data class Pivot(
+ @JsonProperty("titleID") val titleID: Long,
+ @JsonProperty("genreID") val genreID: Long,
+)
+
+data class Vote(
+ @JsonProperty("title_id") val title_id: Long,
+ @JsonProperty("average") val average: String,
+ @JsonProperty("count") val count: Long,
+ @JsonProperty("type") val type: String,
+)
+
+data class VideoElement(
+ @JsonProperty("id") val id: Long,
+ @JsonProperty("slug") val slug: String,
+ @JsonProperty("images") val images: List,
+)
+
+data class Image(
+ @JsonProperty("imageable_id") val imageableID: Long,
+ @JsonProperty("imageable_type") val imageableType: String,
+ @JsonProperty("server_id") val serverID: Long,
+ @JsonProperty("proxy_id") val proxyID: Long,
+ @JsonProperty("url") val url: String,
+ @JsonProperty("type") val type: String,
+// @JsonProperty("sc_url") val scURL: String,
+// @JsonProperty("proxy") val proxy: Proxy,
+// @JsonProperty("server") val server: Proxy
+)
+
+// Proxy is not used and crashes otherwise
+
+//data class Proxy(
+// @JsonProperty("id") val id: Long,
+// @JsonProperty("type") val type: String,
+// @JsonProperty("ip") val ip: String,
+// @JsonProperty("number") val number: Long,
+// @JsonProperty("storage") val storage: Long,
+// @JsonProperty("max_storage") val maxStorage: Long,
+// @JsonProperty("max_conversions") val maxConversions: Any? = null,
+// @JsonProperty("max_publications") val maxPublications: Any? = null,
+// @JsonProperty("created_at") val createdAt: String,
+// @JsonProperty("updated_at") val updatedAt: String,
+// @JsonProperty("upload_bandwidth") val uploadBandwidth: Any? = null,
+// @JsonProperty("upload_bandwidth_limit") val uploadBandwidthLimit: Any? = null
+//)
+
+data class Season(
+ @JsonProperty("id") val id: Long,
+ @JsonProperty("name") val name: String? = "",
+ @JsonProperty("plot") val plot: String? = "",
+ @JsonProperty("date") val date: String? = "",
+ @JsonProperty("number") val number: Long,
+ @JsonProperty("title_id") val title_id: Long,
+ @JsonProperty("createdAt") val createdAt: String? = "",
+ @JsonProperty("updated_at") val updatedAt: String? = "",
+ @JsonProperty("episodes") val episodes: List
+)
+
+data class Episodejson(
+ @JsonProperty("id") val id: Long,
+ @JsonProperty("number") val number: Long,
+ @JsonProperty("name") val name: String? = "",
+ @JsonProperty("plot") val plot: String? = "",
+ @JsonProperty("season_id") val seasonID: Long,
+ @JsonProperty("images") val images: List
+)
+
+data class ImageSeason(
+ @JsonProperty("imageable_id") val imageableID: Long,
+ @JsonProperty("imageable_type") val imageableType: String,
+ @JsonProperty("server_id") val serverID: Long,
+ @JsonProperty("proxy_id") val proxyID: Long,
+ @JsonProperty("url") val url: String,
+ @JsonProperty("type") val type: String,
+ @JsonProperty("original_url") val originalURL: String
+)
+
+data class TrailerElement(
+ @JsonProperty("id") val id: Long? = null,
+ @JsonProperty("url") val url: String? = null,
+ @JsonProperty("host") val host: String? = null,
+ @JsonProperty("videoable_id") val videoableID: Long? = null,
+ @JsonProperty("videoable_type") val videoableType: String? = null,
+ @JsonProperty("created_at") val createdAt: String? = null,
+ @JsonProperty("updated_at") val updatedAt: String? = null,
+ @JsonProperty("size") val size: String? = null,
+ @JsonProperty("created_by") val createdBy: String? = null,
+ @JsonProperty("server_id") val serverID: Long? = null,
+ @JsonProperty("name") val name: String? = null,
+ @JsonProperty("quality") val quality: String? = null,
+ @JsonProperty("original_name") val originalName: Any? = null,
+ @JsonProperty("views") val views: Long? = null,
+ @JsonProperty("public") val public: Long? = null,
+ @JsonProperty("proxy_id") val proxyID: Any? = null,
+ @JsonProperty("proxy_default_id") val proxyDefaultID: Any? = null,
+ @JsonProperty("scws_id") val scwsID: Any? = null
+)
+
+
+class StreamingcommunityProvider : MainAPI() {
+ override var lang = "it"
+ override var mainUrl = "https://streamingcommunity.best"
+ override var name = "Streamingcommunity"
+ override val hasMainPage = true
+ override val hasChromecastSupport = true
+ override val supportedTypes = setOf(
+ TvType.Movie,
+ TvType.TvSeries,
+ )
+
+ private fun translatenumber(num: Int): Int? {
+ return when (num) {
+ 67 -> 1
+ 71 -> 2
+ 72 -> 3
+ 73 -> 4
+ 74 -> 5
+ 75 -> 6
+ 76 -> 7
+ 77 -> 8
+ 78 -> 9
+ 79 -> 10
+ 133 -> 11
+ else -> null
+ }
+ }
+
+ private fun translateip(num: Int): String? {
+ return when (num) {
+ 16 -> "sc-b1-01.scws-content.net"
+ 17 -> "sc-b1-02.scws-content.net"
+ 18 -> "sc-b1-03.scws-content.net"
+ 85 -> "sc-b1-04.scws-content.net"
+ 95 -> "sc-b1-05.scws-content.net"
+ 117 -> "sc-b1-06.scws-content.net"
+ 141 -> "sc-b1-07.scws-content.net"
+ 142 -> "sc-b1-08.scws-content.net"
+ 143 -> "sc-b1-09.scws-content.net"
+ 144 -> "sc-b1-10.scws-content.net"
+ else -> null
+ }
+ }
+
+ companion object {
+ val posterMap = hashMapOf()
+ }
+
+ override suspend fun getMainPage(page: Int, request : MainPageRequest): HomePageResponse {
+ val items = ArrayList()
+ val document = app.get(mainUrl).document
+ document.select("slider-title").subList(0, 3).map { it ->
+ if (it.attr("slider-name") != "In arrivo") {
+ val films = it.attr("titles-json")
+ val lista = mutableListOf()
+ val videoData = parseJson>(films)
+
+ videoData.subList(0, 12).map { searchr ->
+ val id = searchr.id
+ val name = searchr.slug
+ val img = searchr.images[0].url
+ val number = translatenumber(searchr.images[0].serverID.toInt())
+ val ip = translateip(searchr.images[0].proxyID.toInt())
+ val posterurl = "https://$ip/images/$number/$img"
+ val videourl = "$mainUrl/titles/$id-$name"
+ posterMap[videourl] = posterurl
+ val data = app.post("$mainUrl/api/titles/preview/$id", referer = mainUrl).text
+ val datajs = parseJson(data)
+ val type: TvType = if (datajs.type == "movie") {
+ TvType.Movie
+ } else {
+ TvType.TvSeries
+ }
+
+ lista.add(
+ MovieSearchResponse(
+ datajs.name,
+ videourl,
+ this.name,
+ type,
+ posterurl,
+ datajs.releaseDate.substringBefore("-").filter { it.isDigit() }
+ .toIntOrNull(),
+ null,
+ )
+ )
+ }
+ items.add(HomePageList(it.attr("slider-name"), lista))
+ }
+ }
+ if (items.size <= 0) throw ErrorLoadingException()
+ return HomePageResponse(items)
+ }
+
+ override suspend fun search(query: String): List {
+ val queryformatted = query.replace(" ", "%20")
+ val url = "$mainUrl/search?q=$queryformatted"
+ val document = app.get(url).document
+
+ val films =
+ document.selectFirst("the-search-page")!!.attr("records-json").replace(""", """"""")
+
+ val searchresults = parseJson>(films)
+ return searchresults.map { result ->
+ val id = result.id
+ val name = result.slug
+ val img = result.images[0].url
+ val number = translatenumber(result.images[0].serverID.toInt())
+ val ip = translateip(result.images[0].proxyID.toInt())
+ val data = app.post("$mainUrl/api/titles/preview/$id", referer = mainUrl).text
+ val datajs = parseJson(data)
+ val posterurl = "https://$ip/images/$number/$img"
+ val videourl = "$mainUrl/titles/$id-$name"
+ posterMap[videourl] = posterurl
+ if (datajs.type == "movie") {
+ val type = TvType.Movie
+ MovieSearchResponse(
+ datajs.name,
+ videourl,
+ this.name,
+ type,
+ posterurl,
+ datajs.releaseDate.substringBefore("-").filter { it.isDigit() }.toIntOrNull(),
+ null,
+ )
+ } else {
+ val type = TvType.TvSeries
+ TvSeriesSearchResponse(
+ datajs.name,
+ videourl,
+ this.name,
+ type,
+ posterurl,
+ datajs.releaseDate.substringBefore("-").filter { it.isDigit() }.toIntOrNull(),
+ null,
+ )
+ }
+
+ }
+
+ }
+
+ override suspend fun load(url: String): LoadResponse {
+
+ val document = app.get(url).document
+ val poster = posterMap[url]
+ val id = url.substringBefore("-").filter { it.isDigit() }
+ val data = app.post("$mainUrl/api/titles/preview/$id", referer = mainUrl).text
+
+ val datajs = parseJson(data)
+ val type: TvType = if (datajs.type == "movie") {
+ TvType.Movie
+ } else {
+ TvType.TvSeries
+ }
+ val trailerinfojs = document.select("slider-trailer").attr("videos")
+ val trailerinfo = parseJson>(trailerinfojs)
+ val trailerurl: String? = if (trailerinfo.isNotEmpty()) {
+ "https://www.youtube.com/watch?v=${trailerinfo[0].url}"
+ } else {
+ null
+ }
+
+ val year = datajs.releaseDate.substringBefore("-")
+
+ val correlatijs = document.selectFirst("slider-title")!!.attr("titles-json")
+ val listacorr = mutableListOf()
+ val correlatidata = parseJson>(correlatijs)
+ val number : Int = if (correlatidata.size<=15) {correlatidata.size} else correlatidata.size-15
+
+ correlatidata.take(number).map { searchr ->
+ val idcorr = searchr.id
+ val name = searchr.slug
+ val img = searchr.images[0].url
+ val number = translatenumber(searchr.images[0].serverID.toInt())
+ val ip = translateip(searchr.images[0].proxyID.toInt())
+ val datacorrel = app.post("$mainUrl/api/titles/preview/$idcorr", referer = mainUrl).text
+ val datajscorrel = parseJson(datacorrel)
+ val videourl = "$mainUrl/titles/$idcorr-$name"
+ val posterurl = "https://$ip/images/$number/$img"
+
+ posterMap[videourl] = posterurl
+ val typecorr: TvType = if (datajscorrel.type == "movie") {
+ TvType.Movie
+ } else {
+ TvType.TvSeries
+ }
+
+ listacorr.add(
+ MovieSearchResponse(
+ datajscorrel.name,
+ videourl,
+ this.name,
+ typecorr,
+ posterurl,
+ datajscorrel.releaseDate.substringBefore("-").filter { it.isDigit() }
+ .toIntOrNull(),
+ null,
+ )
+ )
+ }
+
+ if (type == TvType.TvSeries) {
+
+ val name = datajs.name
+ val episodeList = arrayListOf()
+
+ val episodes =
+ Html.fromHtml(document.selectFirst("season-select")!!.attr("seasons")).toString()
+ val jsonEpisodes = parseJson>(episodes)
+
+ jsonEpisodes.map { seasons ->
+ val stagione = seasons.number.toInt()
+ val sid = seasons.title_id
+ val episodio = seasons.episodes
+ episodio.map { ep ->
+ val href = "$mainUrl/watch/$sid?e=${ep.id}"
+ val postimage = if (ep.images.isNotEmpty()) {
+ ep.images.first().originalURL
+ } else {
+ ""
+ }
+ episodeList.add(
+
+ newEpisode(href) {
+ this.name = ep.name
+ this.season = stagione
+ this.episode = ep.number.toInt()
+ this.description = ep.plot
+ this.posterUrl = postimage
+ }
+ )
+ }
+ }
+
+
+ if (episodeList.isEmpty()) throw ErrorLoadingException("No Seasons Found")
+
+ return newTvSeriesLoadResponse(name, url, type, episodeList) {
+ this.posterUrl = poster
+ this.year = year.filter { it.isDigit() }.toInt()
+ this.plot = document.selectFirst("div.plot-wrap > p")!!.text()
+ this.duration = datajs.runtime?.toInt()
+ this.rating = (datajs.votes[0].average.toFloatOrNull()?.times(1000))?.toInt()
+ this.tags = datajs.genres.map { it.name }
+ addTrailer(trailerurl)
+ this.recommendations = listacorr
+ }
+
+
+ } else {
+
+ return newMovieLoadResponse(
+ document.selectFirst("div > div > h1")!!.text(),
+ document.select("a.play-hitzone").attr("href"),
+ type,
+ document.select("a.play-hitzone").attr("href")
+ ) {
+ posterUrl = fixUrlNull(poster)
+ this.year = year.filter { it.isDigit() }.toInt()
+ this.plot = document.selectFirst("p.plot")!!.text()
+ this.rating = datajs.votes[0].average.toFloatOrNull()?.times(1000)?.toInt()
+ this.tags = datajs.genres.map { it.name }
+ this.duration = datajs.runtime?.toInt()
+ addTrailer(trailerurl)
+ this.recommendations = listacorr
+ }
+
+ }
+ }
+
+
+ private suspend fun getM3u8Qualities(
+ m3u8Link: String,
+ referer: String,
+ qualityName: String,
+ ): List {
+ return M3u8Helper.generateM3u8(
+ this.name,
+ m3u8Link,
+ referer,
+ name = "${this.name} - $qualityName"
+ )
+ }
+
+
+ override suspend fun loadLinks(
+ data: String,
+ isCasting: Boolean,
+ subtitleCallback: (SubtitleFile) -> Unit,
+ callback: (ExtractorLink) -> Unit
+ ): Boolean {
+ val ip = app.get("https://api.ipify.org/").text
+ val videors = app.get(data).document
+ val scwsidjs = videors.select("video-player").attr("response").replace(""", """"""")
+ val jsn = JSONObject(scwsidjs)
+ val scwsid = jsn.getString("scws_id")
+ val expire = (System.currentTimeMillis() / 1000 + 172800).toString()
+
+ val uno = "$expire$ip Yc8U6r8KjAKAepEA".toByteArray()
+ val due = MessageDigest.getInstance("MD5").digest(uno)
+ val tre = base64Encode(due)
+ val token = tre.replace("=", "").replace("+", "-").replace("/", "_")
+
+
+ val link = "https://scws.xyz/master/$scwsid?token=$token&expires=$expire&n=1&n=1"
+ getM3u8Qualities(link, data, URI(link).host).forEach(callback)
+ return true
+ }
+}
diff --git a/StreamingcommunityProvider/src/main/kotlin/com/lagradost/StreamingcommunityProviderPlugin.kt b/StreamingcommunityProvider/src/main/kotlin/com/lagradost/StreamingcommunityProviderPlugin.kt
new file mode 100644
index 0000000..cf10015
--- /dev/null
+++ b/StreamingcommunityProvider/src/main/kotlin/com/lagradost/StreamingcommunityProviderPlugin.kt
@@ -0,0 +1,14 @@
+
+package com.lagradost
+
+import com.lagradost.cloudstream3.plugins.CloudstreamPlugin
+import com.lagradost.cloudstream3.plugins.Plugin
+import android.content.Context
+
+@CloudstreamPlugin
+class StreamingcommunityProviderPlugin: Plugin() {
+ override fun load(context: Context) {
+ // All providers should be added in this manner. Please don't edit the providers list directly.
+ registerMainAPI(StreamingcommunityProvider())
+ }
+}
\ No newline at end of file
diff --git a/TantiFilmProvider/build.gradle.kts b/TantiFilmProvider/build.gradle.kts
new file mode 100644
index 0000000..9d15f37
--- /dev/null
+++ b/TantiFilmProvider/build.gradle.kts
@@ -0,0 +1,26 @@
+// use an integer for version numbers
+version = 1
+
+
+cloudstream {
+ // 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 // will be 3 if unspecified
+ tvTypes = listOf(
+ "TvSeries",
+ "Movie",
+ )
+
+
+ iconUrl = "https://www.google.com/s2/favicons?domain=tantifilm.autos&sz=24"
+}
\ No newline at end of file
diff --git a/TantiFilmProvider/src/main/AndroidManifest.xml b/TantiFilmProvider/src/main/AndroidManifest.xml
new file mode 100644
index 0000000..29aec9d
--- /dev/null
+++ b/TantiFilmProvider/src/main/AndroidManifest.xml
@@ -0,0 +1,2 @@
+
+
\ No newline at end of file
diff --git a/TantiFilmProvider/src/main/kotlin/com/lagradost/TantiFilmProvider.kt b/TantiFilmProvider/src/main/kotlin/com/lagradost/TantiFilmProvider.kt
new file mode 100644
index 0000000..f030606
--- /dev/null
+++ b/TantiFilmProvider/src/main/kotlin/com/lagradost/TantiFilmProvider.kt
@@ -0,0 +1,234 @@
+package com.lagradost
+
+import com.lagradost.cloudstream3.*
+import com.lagradost.cloudstream3.LoadResponse.Companion.addTrailer
+import com.lagradost.cloudstream3.utils.ExtractorLink
+import com.lagradost.cloudstream3.utils.loadExtractor
+
+
+class TantifilmProvider : MainAPI() {
+ override var lang = "it"
+ override var mainUrl = "https://tantifilm.autos"
+ override var name = "Tantifilm"
+ override val hasMainPage = true
+ override val hasChromecastSupport = true
+ override val supportedTypes = setOf(
+ TvType.Movie,
+ TvType.TvSeries,
+ )
+
+ override val mainPage = mainPageOf(
+ Pair("$mainUrl/watch-genre/al-cinema/page/", "Ultimi Film"),
+ Pair("$mainUrl/serie-tv/page/", "Ultime Serie Tv"),
+ Pair("$mainUrl/watch-genre/film-aggiornati/page/", "Ultimi Film Aggiornati"),
+ )
+
+ override suspend fun getMainPage(
+ page: Int,
+ request : MainPageRequest
+ ): HomePageResponse {
+ val url = request.data + page
+ val soup = app.get(url).document
+ val home = soup.select("div.media3").map {
+ val title = it.selectFirst("p")!!.text().substringBefore("(")
+ val link = it.selectFirst("a")!!.attr("href")
+ TvSeriesSearchResponse(
+ title,
+ link,
+ this.name,
+ TvType.Movie,
+ it.selectFirst("img")!!.attr("src"),
+ null,
+ null,
+ )
+ }
+ return newHomePageResponse(request.name, home)
+ }
+
+ override suspend fun search(query: String): List {
+ val queryformatted = query.replace(" ", "+")
+ val url = "$mainUrl/search/$queryformatted"
+ val doc = app.get(url).document
+ return doc.select("div.film.film-2").map {
+ val href = it.selectFirst("a")!!.attr("href")
+ val poster = it.selectFirst("img")!!.attr("src")
+ val name = it.selectFirst("a > p")!!.text().substringBeforeLast("(")
+ MovieSearchResponse(
+ name,
+ href,
+ this.name,
+ TvType.Movie,
+ poster,
+ null
+ )
+
+ }
+ }
+
+ override suspend fun load(url: String): LoadResponse {
+ val document = app.get(url).document
+ val type = if (document.selectFirst("div.category-film")!!.text().contains("Serie")
+ .not()
+ ) TvType.Movie else TvType.TvSeries
+ val title = document.selectFirst("div.title-film-left")!!.text().substringBefore("(")
+ val descipt = document.select("div.content-left-film > p").map { it.text() }
+ val rating =
+ document.selectFirst("div.star-rating.star-rating-f > span > span")!!
+ .attr("data-rateit-value").toFloatOrNull()
+ ?.times(2857)?.toInt()?.let { minOf(it, 10000) }
+
+ var year = document.selectFirst("div.title-film-left")!!.text().substringAfter("(")
+ .filter { it.isDigit() }
+ year = if (year.length > 4) {
+ year.dropLast(4)
+ } else {
+ year
+ }
+ // ?: does not wor
+ val poster = document.selectFirst("div.image-right-film > img")!!.attr("src")
+
+ val recomm = document.select("div.mediaWrap.mediaWrapAlt.recomended_videos").map {
+ val href = it.selectFirst("a")!!.attr("href")
+ val poster = it.selectFirst("img")!!.attr("src")
+ val name = it.selectFirst("a > p")!!.text().substringBeforeLast("(")
+ MovieSearchResponse(
+ name,
+ href,
+ this.name,
+ TvType.Movie,
+ poster,
+ null
+ )
+
+ }
+
+ val trailerurl = document.selectFirst("#trailer_mob > iframe")!!.attr("src")
+
+ if (type == TvType.TvSeries) {
+ val list = ArrayList>()
+ val urlvideocontainer = document.selectFirst("iframe")!!.attr("src")
+ val videocontainer = app.get(urlvideocontainer).document
+ videocontainer.select("nav.nav1 > select > option").forEach { element ->
+ val season = element.text().toIntOrNull()
+ val href = element.attr("value")
+ if (season != null && season > 0 && !href.isNullOrBlank()) {
+ list.add(Pair(season, fixUrl(href)))
+ }
+ }
+ if (list.isEmpty()) throw ErrorLoadingException("No Seasons Found")
+
+ val episodeList = ArrayList()
+
+ for ((season, seasonurl) in list) {
+ val seasonDocument = app.get(seasonurl).document
+ val episodes = seasonDocument.select("nav.second_nav > select > option")
+ if (episodes.isNotEmpty()) {
+ episodes.forEach { episode ->
+ val href = episode.attr("value")
+ val epNum = episode.text().toIntOrNull()
+ episodeList.add(
+ Episode(
+ href,
+ title,
+ season,
+ epNum,
+ )
+ )
+ }
+ }
+ }
+ return newTvSeriesLoadResponse(
+ title,
+ url,
+ type,
+ episodeList
+ ) {
+ this.posterUrl = fixUrlNull(poster)
+ this.year = year.toIntOrNull()
+ this.plot = descipt[0]
+ this.rating = rating
+ this.recommendations = recomm
+ addTrailer(trailerurl)
+ }
+ } else {
+ val url2 = document.selectFirst("iframe")!!.attr("src")
+ val actorpagelink =
+ document.select("div.content-left-film > p:nth-child(2) > a").attr("href")
+ val actorpagelink2 = document.select("div.content-left-film > p > a").attr("href")
+ val Linkactor: String = actorpagelink.ifEmpty {
+ actorpagelink2
+ }
+
+ val actors: List? = if (Linkactor.isNotEmpty()) {
+ val actorpage = app.get(Linkactor + "cast/").document
+ actorpage.select("article.membro-cast").filter {
+ it -> it.selectFirst("img")
+ ?.attr("src") != "https://www.filmtv.it/imgbank/DUMMY/no_portrait.jpg"
+ }.mapNotNull {
+ val name = it.selectFirst("div.info > h3")!!.text()
+ val image = it.selectFirst("img")?.attr("src")
+ val roleString: String = if (it.selectFirst("h2")?.text() == "Regia") {
+ "Regia"
+ } else {
+ "Attore"
+ }
+ val mainActor = Actor(name, image)
+ ActorData(actor = mainActor, roleString = roleString)
+ }
+ } else {
+ null
+ }
+
+
+ val duratio: Int? = if (descipt.size == 2) {
+ descipt[0].filter { it.isDigit() }.toInt()
+ } else {
+ null
+ }
+ val tags: List? = if (descipt.size == 2) {
+ mutableListOf(descipt[0].substringBefore(" "))
+ } else {
+ null
+ }
+ val plot: String = if (descipt.size == 2) {
+ descipt[1]
+ } else {
+ descipt[0]
+ }
+ return newMovieLoadResponse(
+ title,
+ url2,
+ type,
+ url2
+ ) {
+ posterUrl = fixUrlNull(poster)
+ this.year = year.toIntOrNull()
+ this.plot = plot
+ this.rating = rating
+ this.recommendations = recomm
+ this.tags = tags
+ this.duration = duratio
+ this.actors = actors
+ addTrailer(trailerurl)
+
+ }
+ }
+ }
+
+ override suspend fun loadLinks(
+ data: String,
+ isCasting: Boolean,
+ subtitleCallback: (SubtitleFile) -> Unit,
+ callback: (ExtractorLink) -> Unit
+ ): Boolean {
+ val doc = app.get(data).document
+ val iframe =
+ doc.select("option").map { fixUrl(it.attr("value")) }.filter { it.contains("label") }
+ iframe.forEach { id ->
+ val doc2 = app.get(id).document
+ val id2 = app.get(doc2.selectFirst("iframe")!!.attr("src")).url
+ loadExtractor(id2, data, subtitleCallback, callback)
+ }
+ return true
+ }
+}
\ No newline at end of file
diff --git a/TantiFilmProvider/src/main/kotlin/com/lagradost/TantiFilmProviderPlugin.kt b/TantiFilmProvider/src/main/kotlin/com/lagradost/TantiFilmProviderPlugin.kt
new file mode 100644
index 0000000..0d32f96
--- /dev/null
+++ b/TantiFilmProvider/src/main/kotlin/com/lagradost/TantiFilmProviderPlugin.kt
@@ -0,0 +1,14 @@
+
+package com.lagradost
+
+import com.lagradost.cloudstream3.plugins.CloudstreamPlugin
+import com.lagradost.cloudstream3.plugins.Plugin
+import android.content.Context
+
+@CloudstreamPlugin
+class TantiFilmProviderPlugin: Plugin() {
+ override fun load(context: Context) {
+ // All providers should be added in this manner. Please don't edit the providers list directly.
+ registerMainAPI(TantifilmProvider())
+ }
+}
\ No newline at end of file
diff --git a/settings.gradle.kts b/settings.gradle.kts
index 541da99..ab82ec0 100644
--- a/settings.gradle.kts
+++ b/settings.gradle.kts
@@ -4,18 +4,91 @@ rootProject.name = "CloudstreamPlugins"
// to the includes below.
// Plugins are included like this
-val disabled = listOf()
-
-File(rootDir, ".").eachDir { dir ->
- if (!disabled.contains(dir.name) && File(dir, "build.gradle.kts").exists()) {
- include(dir.name)
- }
-}
-
-fun File.eachDir(block: (File) -> Unit) {
- listFiles()?.filter { it.isDirectory }?.forEach { block(it) }
-}
+include(
+ "XcineProvider",
+ "FilmanProvider",
+ "VMoveeProvider",
+ "AsiaFlixProvider",
+ "DoramasYTProvider",
+// "SflixProProvider",
+ "PinoyMovies",
+ "SeriesflixProvider",
+ "TrailersTwoProvider",
+ "KisskhProvider",
+ "VidstreamBundle",
+ "RebahinProvider",
+ "AllMoviesForYouProvider",
+ "UakinoProvider",
+ "DramaidProvider",
+ "FaselHDProvider",
+ "MeloMovieProvider",
+ "PelisplusProvider",
+ "YomoviesProvider",
+ "AkwamProvider",
+ "IdlixProvider",
+ "NginxProvider",
+ "SoaptwoDayProvider",
+ "PinoyHDXyzProvider",
+ "PelisflixProvider",
+ "SflixProvider",
+ "ElifilmsProvider",
+ "VidSrcProvider",
+ "EgyBestProvider",
+ "VfSerieProvider",
+ "LayarKacaProvider",
+ "EntrepeliculasyseriesProvider",
+ "SuperStream",
+ "FrenchStreamProvider",
+ "MyCimaProvider",
+ "EstrenosDoramasProvider",
+// "HDTodayProvider",
+ "DubokuProvider",
+ "HDMProvider",
+ "PhimmoichillProvider",
+ "OlgplyProvider",
+ "PeliSmartProvider",
+ "VfFilmProvider",
+ "IHaveNoTvProvider",
+ "CuevanaProvider",
+ "CinecalidadProvider",
+ "HDMovie5",
+ "HDrezkaProvider",
+ "PelisplusHDProvider",
+ "MultiplexProvider",
+ //"BflixProvider",
+// "FmoviesToProvider",
+ "TheFlixToProvider",
-// To only include a single project, comment out the previous lines (except the first one), and include your plugin like so:
-// include("PluginName")
+ // <--- Anime providers --->
+ "DubbedAnimeProvider",
+ "TocanimeProvider",
+ "NontonAnimeIDProvider",
+ "WcofunProvider",
+ "TenshiProvider",
+ "KimCartoonProvider",
+ "AllAnimeProvider",
+ "WatchCartoonOnlineProvider",
+ "AnimeFlickProvider",
+ "AnimeSailProvider",
+ "MonoschinosProvider",
+ "AnimefenixProvider",
+ "NeonimeProvider",
+ "KawaiifuProvider",
+ "OploverzProvider",
+ "MundoDonghuaProvider",
+ "NineAnimeProvider",
+ "AnimePaheProvider",
+ "AnimeflvIOProvider",
+ "AniflixProvider",
+ "GogoanimeProvider",
+ "AnimeflvnetProvider",
+ "KuramanimeProvider",
+ "GomunimeProvider",
+ "OtakudesuProvider",
+ "JKAnimeProvider",
+ "AnimekisaProvider",
+ "AnimeIndoProvider",
+ "KuronimeProvider",
+ "EjaTv",
+)