From 3cae0fcc417fc1d08613e9a08febf8f6bac0eded Mon Sep 17 00:00:00 2001 From: CakesTwix Date: Tue, 30 Jul 2024 15:35:30 +0300 Subject: [PATCH] uaflix: Done --- UAFlixProvider/build.gradle.kts | 25 ++ UAFlixProvider/src/main/AndroidManifest.xml | 2 + .../kotlin/com/lagradost/UAFlixProvider.kt | 297 ++++++++++++++++++ .../com/lagradost/UAFlixProviderPlugin.kt | 13 + .../kotlin/com/lagradost/models/PlayerJson.kt | 24 ++ 5 files changed, 361 insertions(+) create mode 100644 UAFlixProvider/build.gradle.kts create mode 100644 UAFlixProvider/src/main/AndroidManifest.xml create mode 100644 UAFlixProvider/src/main/kotlin/com/lagradost/UAFlixProvider.kt create mode 100644 UAFlixProvider/src/main/kotlin/com/lagradost/UAFlixProviderPlugin.kt create mode 100644 UAFlixProvider/src/main/kotlin/com/lagradost/models/PlayerJson.kt diff --git a/UAFlixProvider/build.gradle.kts b/UAFlixProvider/build.gradle.kts new file mode 100644 index 0000000..adcea3b --- /dev/null +++ b/UAFlixProvider/build.gradle.kts @@ -0,0 +1,25 @@ +// use an integer for version numbers +version = 1 + +dependencies { + implementation("com.google.code.gson:gson:2.9.0") +} + +cloudstream { + language = "uk" + // All of these properties are optional, you can safely remove them + + description = "UAFLIX - фільми і серіали NETFLIX українською" + authors = listOf("CakesTwix") + + /** + * Status int as the following: + * 0: Down + * 1: Ok + * 2: Slow + * 3: Beta only + * */ + status = 1 // will be 3 if unspecified + + iconUrl = "https://www.google.com/s2/favicons?domain=uafix.net&sz=%size%" +} diff --git a/UAFlixProvider/src/main/AndroidManifest.xml b/UAFlixProvider/src/main/AndroidManifest.xml new file mode 100644 index 0000000..29aec9d --- /dev/null +++ b/UAFlixProvider/src/main/AndroidManifest.xml @@ -0,0 +1,2 @@ + + \ No newline at end of file diff --git a/UAFlixProvider/src/main/kotlin/com/lagradost/UAFlixProvider.kt b/UAFlixProvider/src/main/kotlin/com/lagradost/UAFlixProvider.kt new file mode 100644 index 0000000..63b2f00 --- /dev/null +++ b/UAFlixProvider/src/main/kotlin/com/lagradost/UAFlixProvider.kt @@ -0,0 +1,297 @@ +package com.lagradost + +import com.lagradost.cloudstream3.AnimeSearchResponse +import com.lagradost.cloudstream3.DubStatus +import com.lagradost.cloudstream3.Episode +import com.lagradost.cloudstream3.HomePageResponse +import com.lagradost.cloudstream3.LoadResponse +import com.lagradost.cloudstream3.LoadResponse.Companion.addActors +import com.lagradost.cloudstream3.MainAPI +import com.lagradost.cloudstream3.MainPageRequest +import com.lagradost.cloudstream3.SearchResponse +import com.lagradost.cloudstream3.SubtitleFile +import com.lagradost.cloudstream3.TvType +import com.lagradost.cloudstream3.addDubStatus +import com.lagradost.cloudstream3.addEpisodes +import com.lagradost.cloudstream3.app +import com.lagradost.cloudstream3.fixUrl +import com.lagradost.cloudstream3.mainPageOf +import com.lagradost.cloudstream3.newAnimeLoadResponse +import com.lagradost.cloudstream3.newAnimeSearchResponse +import com.lagradost.cloudstream3.newHomePageResponse +import com.lagradost.cloudstream3.newMovieLoadResponse +import com.lagradost.cloudstream3.toRatingInt +import com.lagradost.cloudstream3.utils.AppUtils +import com.lagradost.cloudstream3.utils.AppUtils.tryParseJson +import com.lagradost.cloudstream3.utils.ExtractorLink +import com.lagradost.cloudstream3.utils.M3u8Helper +import com.lagradost.models.PlayerJson +import org.jsoup.nodes.Element + +class UAFlixProvider : MainAPI() { + + // Basic Info + override var mainUrl = "https://uafix.net" + override var name = "UAFlix" + override val hasMainPage = true + override var lang = "uk" + override val hasDownloadSupport = true + override val supportedTypes = setOf( + TvType.TvSeries, + TvType.Cartoon, + TvType.Movie, + TvType.Anime + ) + + // Sections + override val mainPage = mainPageOf( + "$mainUrl/serials/page/" to "Серіали", + "$mainUrl/serials/multseial/page/" to "Мультсеріали", + "$mainUrl/cartoons/page/" to "Мультфільми", + "$mainUrl/anime/page/" to "Аніме", + "$mainUrl/dorama/page/" to "Дорами", + "$mainUrl/film/page/" to "Фільми" + ) + + // Main Page + private val animeSelector = ".video-item" + private val titleSelector = ".vi-img" + // private val engTitleSelector = "div.th-title-oname.truncate" + private val hrefSelector = titleSelector + private val posterSelector = ".img-resp-h img" + + // Load info + // private val titleLoadSelector = ".page__subcol-main h1" + // private val genresSelector = "li span:contains(Жанр:) a" + // private val yearSelector = "a[href*=https://uaserials.pro/year/]" + // private val playerSelector = "iframe" + private val descriptionSelector = "div[id=serial-kratko]" + private val ratingSelector = ".mediablock .rat-imdb" + + override suspend fun getMainPage( + page: Int, + request: MainPageRequest + ): HomePageResponse { + val document = app.get(request.data + page).document + + val home = document.select(animeSelector).map { + it.toSearchResponse() + } + return newHomePageResponse(request.name, home) + } + + private fun Element.toSearchResponse(): AnimeSearchResponse { + val title = this.selectFirst(titleSelector)?.attr("alt")?.trim().toString() + // val engTitle = this.selectFirst(engTitleSelector)?.text()?.trim().toString() + val href = this.selectFirst(hrefSelector)?.attr("href").toString() + val posterUrl = fixUrl(this.select(posterSelector).attr("src")) + + return newAnimeSearchResponse(title, href, TvType.Anime) { + // this.otherName = engTitle + this.posterUrl = posterUrl + addDubStatus(isDub = true) + } + + } + + override suspend fun search(query: String): List { + val document = app.post( + url = mainUrl, + data = mapOf( + "do" to "search", + "subaction" to "search", + "story" to query.replace(" ", "+") + ) + ).document + + return document.select(animeSelector).map { + it.toSearchResponse() + } + } + + // Detailed information + override suspend fun load(url: String): LoadResponse { + val document = app.get(url).document + // Parse info + + val title = document.select(".fright h1").text().trim().replace("дивитись онлайн", "") + val engTitle = document.select("span.eng-rus").text() + var poster = fixUrl(document.select(".img-box img").attr("data-src")) + if(poster.isNullOrBlank()){ + poster = fixUrl(document.select(".img-box img").attr("src")) + } + val tags = mutableListOf() + val actors = mutableListOf() + var year = "f".toIntOrNull() + + document.select(".fcols4 .finfo li").forEach { menu -> + with(menu){ + when{ + this.selectFirst("span").text() == "Жанр:" -> menu.select("span[itemprop=genre]").map { tags.add(it.text()) } + this.selectFirst("span").text() == "В ролях:" -> menu.select("span[itemprop=actor]").map { actors.add(it.text()) } + this.selectFirst("span").text() == "Рік виходу:" -> year = menu.select(".year").text().toIntOrNull() + } + } + } + + var tvType = with(url){ + when{ + contains("serials") -> TvType.TvSeries + contains("serials/multseial") -> TvType.Cartoon + contains("film") -> TvType.Movie + contains("cartoons") -> TvType.Movie + contains("anime") -> TvType.Anime + else -> TvType.TvSeries + } + } + val description = document.selectFirst(descriptionSelector)?.text()?.trim() + val rating = document.select(ratingSelector).text().toRatingInt() + + // Parse episodes + val episodes = mutableListOf() + val playerUrl = document.select(".video-box iframe").attr("src") + if(playerUrl.isNullOrBlank()){ // Need parse episode list from site + val pagination = if (document.select(".pagination li").size == 0) 1 else document.select(".pagination li").size + for(i in 1..pagination){ + var episodesList = document + if(i != 1){ + episodesList = app.get("$url?page=$i").document + } + + episodesList.select(".video-item").map { video_item -> + episodes.add( + Episode( + video_item.select(".vi-img").attr("href"), + video_item.select(".vi-rate").text(), + extractIntsFromString(video_item.select(".vi-title").text())[0].value.toIntOrNull(), + extractIntsFromString(video_item.select(".vi-title").text())[1].value.toIntOrNull(), + posterUrl = fixUrl(video_item.select(".img-resp-h img").attr("data-src")) + ) + ) + } + } + + } else { // Player in site + val playerRawJson = app.get(playerUrl).document.select("script").html() + .substringAfterLast("file:\'") + .substringBefore("\',") + + tryParseJson>(playerRawJson)?.map { dubs -> // Dubs + for(season in dubs.folder){ // Seasons + for(episode in season.folder){ // Episodes + episodes.add( + Episode( + "${season.title}, ${episode.title}, $playerUrl", + episode.title, + season.title.replace(" Сезон ","").toIntOrNull(), + episode.title.replace("Серія ","").toIntOrNull(), + episode.poster + ) + ) + } + } + } + } + + + // Parse Episodes as Series + return if (tvType != TvType.Movie) { + + newAnimeLoadResponse(title, url, tvType) { + this.posterUrl = poster + this.engName = engTitle + this.year = year + this.plot = description + this.tags = tags + this.rating = rating + addEpisodes(DubStatus.Dubbed, episodes.sortedBy { it.episode }) + addActors(actors) + } + } else { // Parse as Movie. + + newMovieLoadResponse(title, url, tvType, url) { + this.posterUrl = poster + this.name = engTitle + this.year = year + this.plot = description + this.tags = tags + this.rating = rating + addActors(actors) + } + } + } + + // It works when I click to view the series + override suspend fun loadLinks( + data: String, // (Serial) [Season Index, Episode Name, url] | (Film) [Title, Player Url] + isCasting: Boolean, + subtitleCallback: (SubtitleFile) -> Unit, + callback: (ExtractorLink) -> Unit + ): Boolean { + val dataList = data.split(", ") + + if(dataList.size == 1){ + var playerUrl = app.get(data).document.select(".video-box iframe").attr("src") + if(playerUrl.startsWith("//")){ + playerUrl = "https:$playerUrl" + } + if(playerUrl.contains("/vod/")){ + var playerRawJson = app.get(playerUrl, + headers = mapOf( + "Referer" to "https://uafix.net/", + )).document.select("script").html() + .substringAfterLast("file:\"") + .substringBefore("\",") + + M3u8Helper.generateM3u8( + source = "UAFlix", + streamUrl = playerRawJson, + referer = "https://tortuga.wtf/" + ).forEach(callback) + return true + } + val playerRawJson = app.get(playerUrl).document.select("script").html() + .substringAfterLast("file:\'") + .substringBefore("\',") + tryParseJson>(playerRawJson)?.map { dubs -> // Dubs + for(season in dubs.folder){ // Seasons + // Add as source + M3u8Helper.generateM3u8( + source = dubs.title, + streamUrl = dubs.folder[0].folder[0].file, + referer = "https://tortuga.wtf/" + ).forEach(callback) + } + } + + return true + } + + + val playerRawJson = app.get(dataList[2]).document.select("script").html() + .substringAfterLast("file:\'") + .substringBefore("\',") + tryParseJson>(playerRawJson)?.map { dubs -> // Dubs + for(season in dubs.folder){ // Seasons + if(season.title == dataList[0]){ + for(episode in season.folder){ // Episodes + if(episode.title == dataList[1]){ + // Add as source + M3u8Helper.generateM3u8( + source = dubs.title, + streamUrl = episode.file.replace("https://", "http://"), + referer = "https://tortuga.wtf/" + ).forEach(callback) + } + } + } + } + } + + return true + } + + private fun extractIntsFromString(string: String): List { + return Regex("(\\d+)").findAll(string).toList() + } +} diff --git a/UAFlixProvider/src/main/kotlin/com/lagradost/UAFlixProviderPlugin.kt b/UAFlixProvider/src/main/kotlin/com/lagradost/UAFlixProviderPlugin.kt new file mode 100644 index 0000000..ce19701 --- /dev/null +++ b/UAFlixProvider/src/main/kotlin/com/lagradost/UAFlixProviderPlugin.kt @@ -0,0 +1,13 @@ +package com.lagradost + +import com.lagradost.cloudstream3.plugins.CloudstreamPlugin +import com.lagradost.cloudstream3.plugins.Plugin +import android.content.Context + +@CloudstreamPlugin +class UAFlixProviderPlugin: Plugin() { + override fun load(context: Context) { + // All providers should be added in this manner. Please don't edit the providers list directly. + registerMainAPI(UAFlixProvider()) + } +} diff --git a/UAFlixProvider/src/main/kotlin/com/lagradost/models/PlayerJson.kt b/UAFlixProvider/src/main/kotlin/com/lagradost/models/PlayerJson.kt new file mode 100644 index 0000000..b0cbb53 --- /dev/null +++ b/UAFlixProvider/src/main/kotlin/com/lagradost/models/PlayerJson.kt @@ -0,0 +1,24 @@ +package com.lagradost.models + +import com.google.gson.annotations.SerializedName + +data class PlayerJson ( + + val title : String, + val folder : List +) + +data class Season ( + + val title : String, + val folder : List +) + +data class Episode ( + + val title : String, + val file : String, + val id : String, + val poster : String, + val subtitle : String, +)