mirror of
				https://github.com/hexated/cloudstream-extensions-hexated.git
				synced 2024-08-15 00:03:22 +00:00 
			
		
		
		
	moved providers
This commit is contained in:
		
							parent
							
								
									50024b610c
								
							
						
					
					
						commit
						4f77c2e7d4
					
				
					 115 changed files with 6819 additions and 52 deletions
				
			
		
							
								
								
									
										27
									
								
								AnimeIndoProvider/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								AnimeIndoProvider/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,27 @@ | ||||||
|  | // use an integer for version numbers | ||||||
|  | version = 1 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | cloudstream { | ||||||
|  |     language = "id" | ||||||
|  |     // All of these properties are optional, you can safely remove them | ||||||
|  | 
 | ||||||
|  |     // description = "Lorem Ipsum" | ||||||
|  |      authors = listOf("Hexated") | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Status int as the following: | ||||||
|  |      * 0: Down | ||||||
|  |      * 1: Ok | ||||||
|  |      * 2: Slow | ||||||
|  |      * 3: Beta only | ||||||
|  |      * */ | ||||||
|  |     status = 1 // will be 3 if unspecified | ||||||
|  |     tvTypes = listOf( | ||||||
|  |         "AnimeMovie", | ||||||
|  |         "OVA", | ||||||
|  |         "Anime", | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     iconUrl = "https://www.google.com/s2/favicons?domain=animeindo.sbs&sz=%size%" | ||||||
|  | } | ||||||
							
								
								
									
										2
									
								
								AnimeIndoProvider/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								AnimeIndoProvider/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,2 @@ | ||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <manifest package="com.hexated" /> | ||||||
|  | @ -0,0 +1,192 @@ | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.lagradost.cloudstream3.* | ||||||
|  | import com.lagradost.cloudstream3.APIHolder.getCaptchaToken | ||||||
|  | import com.lagradost.cloudstream3.LoadResponse.Companion.addTrailer | ||||||
|  | import com.lagradost.cloudstream3.utils.ExtractorLink | ||||||
|  | import com.lagradost.cloudstream3.utils.loadExtractor | ||||||
|  | import com.lagradost.nicehttp.NiceResponse | ||||||
|  | import org.jsoup.Jsoup | ||||||
|  | import org.jsoup.nodes.Element | ||||||
|  | 
 | ||||||
|  | class AnimeIndoProvider : MainAPI() { | ||||||
|  |     override var mainUrl = "https://animeindo.sbs" | ||||||
|  |     override var name = "AnimeIndo" | ||||||
|  |     override val hasMainPage = true | ||||||
|  |     override var lang = "id" | ||||||
|  |     override val hasDownloadSupport = true | ||||||
|  | 
 | ||||||
|  |     override val supportedTypes = setOf( | ||||||
|  |         TvType.Anime, | ||||||
|  |         TvType.AnimeMovie, | ||||||
|  |         TvType.OVA | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     companion object { | ||||||
|  |         fun getType(t: String): TvType { | ||||||
|  |             return if (t.contains("OVA") || t.contains("Special")) TvType.OVA | ||||||
|  |             else if (t.contains("Movie")) TvType.AnimeMovie | ||||||
|  |             else TvType.Anime | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         fun getStatus(t: String): ShowStatus { | ||||||
|  |             return when (t) { | ||||||
|  |                 "Finished Airing" -> ShowStatus.Completed | ||||||
|  |                 "Currently Airing" -> ShowStatus.Ongoing | ||||||
|  |                 else -> ShowStatus.Completed | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         private suspend fun request(url: String): NiceResponse { | ||||||
|  |             val req = app.get( | ||||||
|  |                 url, | ||||||
|  |                 cookies = mapOf("recaptcha_cookie" to "#Asia/Jakarta#-420#win32#Windows#0,false,false#Google Inc. (Intel)~ANGLE (Intel, Intel(R) HD Graphics 400 Direct3D11 vs_5_0 ps_5_0)") | ||||||
|  |             ) | ||||||
|  |             if (req.isSuccessful) { | ||||||
|  |                 return req | ||||||
|  |             } else { | ||||||
|  |                 val document = app.get(url).document | ||||||
|  |                 val captchaKey = | ||||||
|  |                     document.select("script[src*=https://www.google.com/recaptcha/api.js?render=]") | ||||||
|  |                         .attr("src").substringAfter("render=").substringBefore("&") | ||||||
|  |                 val token = getCaptchaToken(url, captchaKey) | ||||||
|  |                 return app.post( | ||||||
|  |                     url, | ||||||
|  |                     data = mapOf( | ||||||
|  |                         "action" to "recaptcha_for_all", | ||||||
|  |                         "token" to "$token", | ||||||
|  |                         "sitekey" to captchaKey | ||||||
|  |                     ) | ||||||
|  |                 ) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override val mainPage = mainPageOf( | ||||||
|  |         "$mainUrl/anime-terbaru/page/" to "Anime Terbaru", | ||||||
|  |         "$mainUrl/donghua-terbaru/page/" to "Donghua Terbaru" | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     override suspend fun getMainPage( | ||||||
|  |         page: Int, | ||||||
|  |         request: MainPageRequest | ||||||
|  |     ): HomePageResponse { | ||||||
|  |         val document = request(request.data + page).document | ||||||
|  |         val home = document.select("div.post-show > article").mapNotNull { | ||||||
|  |             it.toSearchResult() | ||||||
|  |         } | ||||||
|  |         return newHomePageResponse(request.name, home) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun getProperAnimeLink(uri: String): String { | ||||||
|  |         return if (uri.contains("/anime/")) { | ||||||
|  |             uri | ||||||
|  |         } else { | ||||||
|  |             var title = uri.substringAfter("$mainUrl/") | ||||||
|  |             title = when { | ||||||
|  |                 (title.contains("-episode")) && !(title.contains("-movie")) -> Regex("(.+)-episode").find( | ||||||
|  |                     title | ||||||
|  |                 )?.groupValues?.get(1).toString() | ||||||
|  |                 (title.contains("-movie")) -> Regex("(.+)-movie").find(title)?.groupValues?.get( | ||||||
|  |                     1 | ||||||
|  |                 ).toString() | ||||||
|  |                 else -> title | ||||||
|  |             } | ||||||
|  |             "$mainUrl/anime/$title" | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun Element.toSearchResult(): AnimeSearchResponse? { | ||||||
|  |         val title = this.selectFirst("div.title")?.text()?.trim() ?: return null | ||||||
|  |         val href = getProperAnimeLink(this.selectFirst("a")!!.attr("href")) | ||||||
|  |         val posterUrl = this.select("img[itemprop=image]").attr("src").toString() | ||||||
|  |         val type = getType(this.select("div.type").text().trim()) | ||||||
|  |         val epNum = | ||||||
|  |             this.selectFirst("span.episode")?.ownText()?.replace(Regex("[^0-9]"), "")?.trim() | ||||||
|  |                 ?.toIntOrNull() | ||||||
|  |         return newAnimeSearchResponse(title, href, type) { | ||||||
|  |             this.posterUrl = posterUrl | ||||||
|  |             addSub(epNum) | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun search(query: String): List<SearchResponse> { | ||||||
|  |         val link = "$mainUrl/?s=$query" | ||||||
|  |         val document = request(link).document | ||||||
|  | 
 | ||||||
|  |         return document.select(".site-main.relat > article").map { | ||||||
|  |             val title = it.selectFirst("div.title > h2")!!.ownText().trim() | ||||||
|  |             val href = it.selectFirst("a")!!.attr("href") | ||||||
|  |             val posterUrl = it.selectFirst("img")!!.attr("src").toString() | ||||||
|  |             val type = getType(it.select("div.type").text().trim()) | ||||||
|  |             newAnimeSearchResponse(title, href, type) { | ||||||
|  |                 this.posterUrl = posterUrl | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun load(url: String): LoadResponse { | ||||||
|  |         val document = request(url).document | ||||||
|  | 
 | ||||||
|  |         val title = document.selectFirst("h1.entry-title")?.text().toString().trim() | ||||||
|  |         val poster = document.selectFirst("div.thumb > img[itemprop=image]")?.attr("src") | ||||||
|  |         val tags = document.select("div.genxed > a").map { it.text() } | ||||||
|  |         val type = getType( | ||||||
|  |             document.selectFirst("div.info-content > div.spe > span:nth-child(6)")?.ownText() | ||||||
|  |                 .toString() | ||||||
|  |         ) | ||||||
|  |         val year = Regex("\\d, ([0-9]*)").find( | ||||||
|  |             document.select("div.info-content > div.spe > span:nth-child(9) > time").text() | ||||||
|  |         )?.groupValues?.get(1)?.toIntOrNull() | ||||||
|  |         val status = getStatus( | ||||||
|  |             document.selectFirst("div.info-content > div.spe > span:nth-child(1)")!!.ownText() | ||||||
|  |                 .trim() | ||||||
|  |         ) | ||||||
|  |         val description = document.select("div[itemprop=description] > p").text() | ||||||
|  |         val trailer = document.selectFirst("div.player-embed iframe")?.attr("src") | ||||||
|  |         val episodes = document.select("div.lstepsiode.listeps ul li").mapNotNull { | ||||||
|  |             val header = it.selectFirst("span.lchx > a") ?: return@mapNotNull null | ||||||
|  |             val name = header.text().trim() | ||||||
|  |             val episode = header.text().trim().replace("Episode", "").trim().toIntOrNull() | ||||||
|  |             val link = fixUrl(header.attr("href")) | ||||||
|  |             Episode(link, name = name, episode = episode) | ||||||
|  |         }.reversed() | ||||||
|  | 
 | ||||||
|  |         return newAnimeLoadResponse(title, url, type) { | ||||||
|  |             engName = title | ||||||
|  |             posterUrl = poster | ||||||
|  |             this.year = year | ||||||
|  |             addEpisodes(DubStatus.Subbed, episodes) | ||||||
|  |             showStatus = status | ||||||
|  |             plot = description | ||||||
|  |             this.tags = tags | ||||||
|  |             addTrailer(trailer) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun loadLinks( | ||||||
|  |         data: String, | ||||||
|  |         isCasting: Boolean, | ||||||
|  |         subtitleCallback: (SubtitleFile) -> Unit, | ||||||
|  |         callback: (ExtractorLink) -> Unit | ||||||
|  |     ): Boolean { | ||||||
|  | 
 | ||||||
|  |         val document = request(data).document | ||||||
|  |         document.select("div.itemleft > .mirror > option").mapNotNull { | ||||||
|  |             fixUrl(Jsoup.parse(base64Decode(it.attr("value"))).select("iframe").attr("src")) | ||||||
|  |         }.apmap { | ||||||
|  |             if (it.startsWith("https://uservideo.xyz")) { | ||||||
|  |                 app.get(it, referer = "$mainUrl/").document.select("iframe").attr("src") | ||||||
|  |             } else { | ||||||
|  |                 it | ||||||
|  |             } | ||||||
|  |         }.apmap { | ||||||
|  |             loadExtractor(it, data, subtitleCallback, callback) | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return true | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | @ -0,0 +1,14 @@ | ||||||
|  | 
 | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.lagradost.cloudstream3.plugins.CloudstreamPlugin | ||||||
|  | import com.lagradost.cloudstream3.plugins.Plugin | ||||||
|  | import android.content.Context | ||||||
|  | 
 | ||||||
|  | @CloudstreamPlugin | ||||||
|  | class AnimeIndoProviderPlugin: Plugin() { | ||||||
|  |     override fun load(context: Context) { | ||||||
|  |         // All providers should be added in this manner. Please don't edit the providers list directly. | ||||||
|  |         registerMainAPI(AnimeIndoProvider()) | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										27
									
								
								AnimeSailProvider/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								AnimeSailProvider/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,27 @@ | ||||||
|  | // use an integer for version numbers | ||||||
|  | version = 2 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | cloudstream { | ||||||
|  |     language = "id" | ||||||
|  |     // All of these properties are optional, you can safely remove them | ||||||
|  | 
 | ||||||
|  |     // description = "Lorem Ipsum" | ||||||
|  |      authors = listOf("Hexated") | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * 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=111.90.143.42&sz=%size%" | ||||||
|  | } | ||||||
							
								
								
									
										2
									
								
								AnimeSailProvider/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								AnimeSailProvider/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,2 @@ | ||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <manifest package="com.hexated" /> | ||||||
|  | @ -0,0 +1,195 @@ | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.lagradost.cloudstream3.* | ||||||
|  | import com.lagradost.cloudstream3.mvvm.safeApiCall | ||||||
|  | import com.lagradost.cloudstream3.utils.ExtractorLink | ||||||
|  | import com.lagradost.cloudstream3.utils.Qualities | ||||||
|  | import com.lagradost.cloudstream3.utils.loadExtractor | ||||||
|  | import com.lagradost.nicehttp.NiceResponse | ||||||
|  | import org.jsoup.Jsoup | ||||||
|  | import org.jsoup.nodes.Element | ||||||
|  | 
 | ||||||
|  | class AnimeSailProvider : MainAPI() { | ||||||
|  |     override var mainUrl = "https://111.90.143.42" | ||||||
|  |     override var name = "AnimeSail" | ||||||
|  |     override val hasMainPage = true | ||||||
|  |     override var lang = "id" | ||||||
|  |     override val hasDownloadSupport = true | ||||||
|  | 
 | ||||||
|  |     override val supportedTypes = setOf( | ||||||
|  |         TvType.Anime, | ||||||
|  |         TvType.AnimeMovie, | ||||||
|  |         TvType.OVA | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     companion object { | ||||||
|  |         fun getType(t: String): TvType { | ||||||
|  |             return if (t.contains("OVA") || t.contains("Special")) TvType.OVA | ||||||
|  |             else if (t.contains("Movie")) TvType.AnimeMovie | ||||||
|  |             else TvType.Anime | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         fun getStatus(t: String): ShowStatus { | ||||||
|  |             return when (t) { | ||||||
|  |                 "Completed" -> ShowStatus.Completed | ||||||
|  |                 "Ongoing" -> ShowStatus.Ongoing | ||||||
|  |                 else -> ShowStatus.Completed | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private suspend fun request(url: String, ref: String? = null): NiceResponse { | ||||||
|  |         return app.get( | ||||||
|  |             url, | ||||||
|  |             headers = mapOf("Accept" to "text/html,application/xhtml+xml,application/xml;q=0.9,image/avif,image/webp,*/*;q=0.8"), | ||||||
|  |             cookies = mapOf("_as_ipin_ct" to "ID"), | ||||||
|  |             referer = ref | ||||||
|  |         ) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override val mainPage = mainPageOf( | ||||||
|  |         "$mainUrl/page/" to "Episode Terbaru", | ||||||
|  |         "$mainUrl/movie-terbaru/page/" to "Movie Terbaru", | ||||||
|  |         "$mainUrl/genres/donghua/page/" to "Donghua" | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     override suspend fun getMainPage(page: Int, request: MainPageRequest): HomePageResponse { | ||||||
|  |         val document = request(request.data + page).document | ||||||
|  |         val home = document.select("article").map { | ||||||
|  |             it.toSearchResult() | ||||||
|  |         } | ||||||
|  |         return newHomePageResponse(request.name, home) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun getProperAnimeLink(uri: String): String { | ||||||
|  |         return if (uri.contains("/anime/")) { | ||||||
|  |             uri | ||||||
|  |         } else { | ||||||
|  |             var title = uri.substringAfter("$mainUrl/") | ||||||
|  |             title = when { | ||||||
|  |                 (title.contains("-episode")) && !(title.contains("-movie")) -> title.substringBefore( | ||||||
|  |                     "-episode" | ||||||
|  |                 ) | ||||||
|  |                 (title.contains("-movie")) -> title.substringBefore("-movie") | ||||||
|  |                 else -> title | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             "$mainUrl/anime/$title" | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun Element.toSearchResult(): AnimeSearchResponse { | ||||||
|  |         val href = getProperAnimeLink(fixUrlNull(this.selectFirst("a")?.attr("href")).toString()) | ||||||
|  |         val title = this.select(".tt > h2").text().trim() | ||||||
|  |         val posterUrl = fixUrlNull(this.selectFirst("div.limit img")?.attr("src")) | ||||||
|  |         val epNum = this.selectFirst(".tt > h2")?.text()?.let { | ||||||
|  |             Regex("Episode\\s?([0-9]+)").find(it)?.groupValues?.getOrNull(1)?.toIntOrNull() | ||||||
|  |         } | ||||||
|  |         return newAnimeSearchResponse(title, href, TvType.Anime) { | ||||||
|  |             this.posterUrl = posterUrl | ||||||
|  |             addSub(epNum) | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun search(query: String): List<SearchResponse> { | ||||||
|  |         val link = "$mainUrl/?s=$query" | ||||||
|  |         val document = request(link).document | ||||||
|  | 
 | ||||||
|  |         return document.select("div.listupd article").map { | ||||||
|  |             it.toSearchResult() | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun load(url: String): LoadResponse { | ||||||
|  |         val document = request(url).document | ||||||
|  | 
 | ||||||
|  |         val title = document.selectFirst("h1.entry-title")?.text().toString().trim() | ||||||
|  |         val type = getType( | ||||||
|  |             document.select("tbody th:contains(Tipe)").next().text() | ||||||
|  |         ) | ||||||
|  |         val episodes = document.select("ul.daftar > li").map { | ||||||
|  |             val header = it.select("a").text().trim() | ||||||
|  |             val name = | ||||||
|  |                 Regex("(Episode\\s?[0-9]+)").find(header)?.groupValues?.getOrNull(0) ?: header | ||||||
|  |             val link = fixUrl(it.select("a").attr("href")) | ||||||
|  |             Episode(link, name = name) | ||||||
|  |         }.reversed() | ||||||
|  | 
 | ||||||
|  |         return newAnimeLoadResponse(title, url, type) { | ||||||
|  |             posterUrl = document.selectFirst("div.entry-content > img")?.attr("src") | ||||||
|  |             this.year = | ||||||
|  |                 document.select("tbody th:contains(Dirilis)").next().text().trim().toIntOrNull() | ||||||
|  |             addEpisodes(DubStatus.Subbed, episodes) | ||||||
|  |             showStatus = | ||||||
|  |                 getStatus(document.select("tbody th:contains(Status)").next().text().trim()) | ||||||
|  |             plot = document.selectFirst("div.entry-content > p")?.text() | ||||||
|  |             this.tags = | ||||||
|  |                 document.select("tbody th:contains(Genre)").next().select("a").map { it.text() } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun loadLinks( | ||||||
|  |         data: String, | ||||||
|  |         isCasting: Boolean, | ||||||
|  |         subtitleCallback: (SubtitleFile) -> Unit, | ||||||
|  |         callback: (ExtractorLink) -> Unit | ||||||
|  |     ): Boolean { | ||||||
|  | 
 | ||||||
|  |         val document = request(data).document | ||||||
|  | 
 | ||||||
|  |         document.select(".mobius > .mirror > option").apmap { | ||||||
|  |             safeApiCall { | ||||||
|  |                 val iframe = fixUrl( | ||||||
|  |                     Jsoup.parse(base64Decode(it.attr("data-em"))).select("iframe").attr("src") | ||||||
|  |                         ?: throw ErrorLoadingException("No iframe found") | ||||||
|  |                 ) | ||||||
|  | 
 | ||||||
|  |                 when { | ||||||
|  |                     iframe.startsWith("$mainUrl/utils/player/arch/") || iframe.startsWith( | ||||||
|  |                         "$mainUrl/utils/player/race/" | ||||||
|  |                     ) -> request(iframe, ref = data).document.select("source").attr("src") | ||||||
|  |                         .let { link -> | ||||||
|  |                             val source = | ||||||
|  |                                 when { | ||||||
|  |                                     iframe.contains("/arch/") -> "Arch" | ||||||
|  |                                     iframe.contains("/race/") -> "Race" | ||||||
|  |                                     else -> this.name | ||||||
|  |                                 } | ||||||
|  |                             val quality = | ||||||
|  |                                 Regex("\\.([0-9]{3,4})\\.").find(link)?.groupValues?.get(1) | ||||||
|  |                             callback.invoke( | ||||||
|  |                                 ExtractorLink( | ||||||
|  |                                     source = source, | ||||||
|  |                                     name = source, | ||||||
|  |                                     url = link, | ||||||
|  |                                     referer = mainUrl, | ||||||
|  |                                     quality = quality?.toIntOrNull() ?: Qualities.Unknown.value | ||||||
|  |                                 ) | ||||||
|  |                             ) | ||||||
|  |                         } | ||||||
|  | //                    skip for now | ||||||
|  | //                    iframe.startsWith("$mainUrl/utils/player/fichan/") -> "" | ||||||
|  | //                    iframe.startsWith("$mainUrl/utils/player/blogger/") -> "" | ||||||
|  |                     iframe.startsWith("https://aghanim.xyz/tools/redirect/") -> { | ||||||
|  |                         val link = "https://rasa-cintaku-semakin-berantai.xyz/v/${iframe.substringAfter("id=").substringBefore("&token")}" | ||||||
|  |                         loadExtractor(link, mainUrl, subtitleCallback, callback) | ||||||
|  |                     } | ||||||
|  |                     iframe.startsWith("$mainUrl/utils/player/framezilla/") || iframe.startsWith("https://uservideo.xyz") -> { | ||||||
|  |                         request(iframe, ref = data).document.select("iframe").attr("src") | ||||||
|  |                             .let { link -> | ||||||
|  |                                 loadExtractor(fixUrl(link), mainUrl, subtitleCallback, callback) | ||||||
|  |                             } | ||||||
|  |                     } | ||||||
|  |                     else -> { | ||||||
|  |                         loadExtractor(iframe, mainUrl, subtitleCallback, callback) | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return true | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | @ -0,0 +1,14 @@ | ||||||
|  | 
 | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.lagradost.cloudstream3.plugins.CloudstreamPlugin | ||||||
|  | import com.lagradost.cloudstream3.plugins.Plugin | ||||||
|  | import android.content.Context | ||||||
|  | 
 | ||||||
|  | @CloudstreamPlugin | ||||||
|  | class AnimeSailProviderPlugin: Plugin() { | ||||||
|  |     override fun load(context: Context) { | ||||||
|  |         // All providers should be added in this manner. Please don't edit the providers list directly. | ||||||
|  |         registerMainAPI(AnimeSailProvider()) | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										27
									
								
								Anizm/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								Anizm/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,27 @@ | ||||||
|  | // use an integer for version numbers | ||||||
|  | version = 1 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | cloudstream { | ||||||
|  |     language = "tr" | ||||||
|  |     // All of these properties are optional, you can safely remove them | ||||||
|  | 
 | ||||||
|  |     // description = "Lorem Ipsum" | ||||||
|  |      authors = listOf("Hexated") | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * 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=anizm.net&sz=%size%" | ||||||
|  | } | ||||||
							
								
								
									
										2
									
								
								Anizm/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								Anizm/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,2 @@ | ||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <manifest package="com.hexated" /> | ||||||
							
								
								
									
										195
									
								
								Anizm/src/main/kotlin/com/hexated/Anizm.kt
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										195
									
								
								Anizm/src/main/kotlin/com/hexated/Anizm.kt
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,195 @@ | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import android.util.Log | ||||||
|  | import com.fasterxml.jackson.annotation.JsonProperty | ||||||
|  | import com.lagradost.cloudstream3.* | ||||||
|  | import com.lagradost.cloudstream3.LoadResponse.Companion.addTrailer | ||||||
|  | import com.lagradost.cloudstream3.mvvm.safeApiCall | ||||||
|  | import com.lagradost.cloudstream3.utils.* | ||||||
|  | import org.jsoup.Jsoup | ||||||
|  | import org.jsoup.nodes.Element | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | class Anizm : MainAPI() { | ||||||
|  |     override var mainUrl = "https://anizm.net" | ||||||
|  |     override var name = "Anizm" | ||||||
|  |     override val hasMainPage = true | ||||||
|  |     override var lang = "tr" | ||||||
|  |     override val hasDownloadSupport = true | ||||||
|  | 
 | ||||||
|  |     override val supportedTypes = setOf( | ||||||
|  |         TvType.Anime, | ||||||
|  |         TvType.AnimeMovie, | ||||||
|  |         TvType.OVA | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     companion object { | ||||||
|  |         private const val mainServer = "https://anizmplayer.com" | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override val mainPage = mainPageOf( | ||||||
|  |         "$mainUrl/anime-izle?sayfa=" to "Son Eklenen Animeler", | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     override suspend fun getMainPage( | ||||||
|  |         page: Int, | ||||||
|  |         request: MainPageRequest | ||||||
|  |     ): HomePageResponse { | ||||||
|  |         val document = app.get(request.data + page).document | ||||||
|  |         val home = document.select("div.restrictedWidth div#episodesMiddle").mapNotNull { | ||||||
|  |             it.toSearchResult() | ||||||
|  |         } | ||||||
|  |         return newHomePageResponse(request.name, home) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun getProperAnimeLink(uri: String): String { | ||||||
|  |         return if (uri.contains("-bolum")) { | ||||||
|  |             "$mainUrl/${uri.substringAfter("$mainUrl/").replace(Regex("-[0-9]+-bolum.*"), "")}" | ||||||
|  |         } else { | ||||||
|  |             uri | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun Element.toSearchResult(): AnimeSearchResponse? { | ||||||
|  |         val href = getProperAnimeLink(this.selectFirst("a")!!.attr("href")) | ||||||
|  |         val title = this.selectFirst("div.title, h5.animeTitle a")?.text() ?: return null | ||||||
|  |         val posterUrl = fixUrlNull(this.selectFirst("img")?.attr("src")) | ||||||
|  |         val episode = this.selectFirst("div.truncateText")?.text()?.let { | ||||||
|  |             Regex("([0-9]+).\\s?Bölüm").find(it)?.groupValues?.getOrNull(1)?.toIntOrNull() | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return newAnimeSearchResponse(title, href, TvType.Anime) { | ||||||
|  |             this.posterUrl = posterUrl | ||||||
|  |             addSub(episode) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun search(query: String): List<SearchResponse> { | ||||||
|  |         val document = app.get( | ||||||
|  |             "$mainUrl/fullViewSearch?search=$query&skip=0", | ||||||
|  |             headers = mapOf("X-Requested-With" to "XMLHttpRequest") | ||||||
|  |         ).document | ||||||
|  | 
 | ||||||
|  |         return document.select("div.searchResultItem").mapNotNull { | ||||||
|  |             it.toSearchResult() | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun load(url: String): LoadResponse { | ||||||
|  |         val document = app.get(url).document | ||||||
|  | 
 | ||||||
|  |         val title = document.selectFirst("h2.anizm_pageTitle a")!!.text().trim() | ||||||
|  |         val type = | ||||||
|  |             if (document.select("div.ui.grid div.four.wide").size == 1) TvType.Movie else TvType.Anime | ||||||
|  |         val trailer = document.select("div.yt-hd-thumbnail-inner-container iframe").attr("src") | ||||||
|  |         val episodes = document.select("div.ui.grid div.four.wide").map { | ||||||
|  |             val name = it.select("div.episodeBlock").text() | ||||||
|  |             val link = fixUrl(it.selectFirst("a")?.attr("href").toString()) | ||||||
|  |             Episode(link, name) | ||||||
|  |         } | ||||||
|  |         return newAnimeLoadResponse(title, url, type) { | ||||||
|  |             posterUrl = fixUrlNull(document.selectFirst("div.infoPosterImg > img")?.attr("src")) | ||||||
|  |             this.year = document.select("div.infoSta ul li:first-child").text().trim().toIntOrNull() | ||||||
|  |             addEpisodes(DubStatus.Subbed, episodes) | ||||||
|  |             plot = document.select("div.infoDesc").text().trim() | ||||||
|  |             this.tags = document.select("span.dataValue span.ui.label").map { it.text() } | ||||||
|  |             addTrailer(trailer) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private suspend fun invokeLokalSource( | ||||||
|  |         url: String, | ||||||
|  |         translator: String, | ||||||
|  |         sourceCallback: (ExtractorLink) -> Unit | ||||||
|  |     ) { | ||||||
|  |         app.get(url, referer = "$mainUrl/").document.select("script").find { script -> | ||||||
|  |             script.data().contains("eval(function(p,a,c,k,e,d)") | ||||||
|  |         }?.let { | ||||||
|  |             val key = getAndUnpack(it.data()).substringAfter("FirePlayer(\"").substringBefore("\",") | ||||||
|  |             val referer = "$mainServer/video/$key" | ||||||
|  |             val link = "$mainServer/player/index.php?data=$key&do=getVideo" | ||||||
|  |             Log.i("hexated", link) | ||||||
|  |             app.post( | ||||||
|  |                 link, | ||||||
|  |                 data = mapOf("hash" to key, "r" to "$mainUrl/"), | ||||||
|  |                 referer = referer, | ||||||
|  |                 headers = mapOf( | ||||||
|  |                     "Accept" to "*/*", | ||||||
|  |                     "Origin" to mainServer, | ||||||
|  |                     "Content-Type" to "application/x-www-form-urlencoded; charset=UTF-8", | ||||||
|  |                     "X-Requested-With" to "XMLHttpRequest" | ||||||
|  |                 ) | ||||||
|  |             ).parsedSafe<Source>()?.videoSource?.let { m3uLink -> | ||||||
|  |                 M3u8Helper.generateM3u8( | ||||||
|  |                     "${this.name} ($translator)", | ||||||
|  |                     m3uLink, | ||||||
|  |                     referer | ||||||
|  |                 ).forEach(sourceCallback) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun loadLinks( | ||||||
|  |         data: String, | ||||||
|  |         isCasting: Boolean, | ||||||
|  |         subtitleCallback: (SubtitleFile) -> Unit, | ||||||
|  |         callback: (ExtractorLink) -> Unit | ||||||
|  |     ): Boolean { | ||||||
|  |         val document = app.get(data).document | ||||||
|  |         document.select("div.episodeTranslators div#fansec").map { | ||||||
|  |             Pair(it.select("a").attr("translator"), it.select("div.title").text()) | ||||||
|  |         }.apmap { (url, translator) -> | ||||||
|  |             safeApiCall { | ||||||
|  |                 app.get( | ||||||
|  |                     url, | ||||||
|  |                     referer = data, | ||||||
|  |                     headers = mapOf( | ||||||
|  |                         "Accept" to "application/json, text/javascript, */*; q=0.01", | ||||||
|  |                         "X-Requested-With" to "XMLHttpRequest" | ||||||
|  |                     ) | ||||||
|  |                 ).parsedSafe<Translators>()?.data?.let { | ||||||
|  |                     Jsoup.parse(it).select("a").apmap { video -> | ||||||
|  |                         app.get( | ||||||
|  |                             video.attr("video"), | ||||||
|  |                             referer = data, | ||||||
|  |                             headers = mapOf( | ||||||
|  |                                 "Accept" to "application/json, text/javascript, */*; q=0.01", | ||||||
|  |                                 "X-Requested-With" to "XMLHttpRequest" | ||||||
|  |                             ) | ||||||
|  |                         ).parsedSafe<Videos>()?.player?.let { iframe -> | ||||||
|  |                             Jsoup.parse(iframe).select("iframe").attr("src").let { link -> | ||||||
|  |                                 when { | ||||||
|  |                                     link.startsWith(mainServer) -> { | ||||||
|  |                                         invokeLokalSource(link, translator, callback) | ||||||
|  |                                     } | ||||||
|  |                                     else -> { | ||||||
|  |                                         loadExtractor( | ||||||
|  |                                             fixUrl(link), | ||||||
|  |                                             "$mainUrl/", | ||||||
|  |                                             subtitleCallback, | ||||||
|  |                                             callback | ||||||
|  |                                         ) | ||||||
|  |                                     } | ||||||
|  |                                 } | ||||||
|  |                             } | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return true | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     data class Source( | ||||||
|  |         @JsonProperty("videoSource") val videoSource: String?, | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     data class Videos( | ||||||
|  |         @JsonProperty("player") val player: String?, | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     data class Translators( | ||||||
|  |         @JsonProperty("data") val data: String?, | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | @ -1,13 +1,14 @@ | ||||||
| package com.example | 
 | ||||||
|  | package com.hexated | ||||||
| 
 | 
 | ||||||
| import com.lagradost.cloudstream3.plugins.CloudstreamPlugin | import com.lagradost.cloudstream3.plugins.CloudstreamPlugin | ||||||
| import com.lagradost.cloudstream3.plugins.Plugin | import com.lagradost.cloudstream3.plugins.Plugin | ||||||
| import android.content.Context | import android.content.Context | ||||||
| 
 | 
 | ||||||
| @CloudstreamPlugin | @CloudstreamPlugin | ||||||
| class TestPlugin: Plugin() { | class AnizmPlugin: Plugin() { | ||||||
|     override fun load(context: Context) { |     override fun load(context: Context) { | ||||||
|         // All providers should be added in this manner. Please don't edit the providers list directly. |         // All providers should be added in this manner. Please don't edit the providers list directly. | ||||||
|         registerMainAPI(ExampleProvider()) |         registerMainAPI(Anizm()) | ||||||
|     } |     } | ||||||
| } | } | ||||||
							
								
								
									
										26
									
								
								DramaidProvider/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								DramaidProvider/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,26 @@ | ||||||
|  | // use an integer for version numbers | ||||||
|  | version = 2 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | cloudstream { | ||||||
|  |     language = "id" | ||||||
|  |     // All of these properties are optional, you can safely remove them | ||||||
|  | 
 | ||||||
|  |     // description = "Lorem Ipsum" | ||||||
|  |      authors = listOf("Hexated") | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Status int as the following: | ||||||
|  |      * 0: Down | ||||||
|  |      * 1: Ok | ||||||
|  |      * 2: Slow | ||||||
|  |      * 3: Beta only | ||||||
|  |      * */ | ||||||
|  |     status = 1 // will be 3 if unspecified | ||||||
|  |     tvTypes = listOf( | ||||||
|  |         "AsianDrama", | ||||||
|  |         "Movie", | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     iconUrl = "https://www.google.com/s2/favicons?domain=dramaid.asia&sz=%size%" | ||||||
|  | } | ||||||
|  | @ -1,2 +1,2 @@ | ||||||
| <?xml version="1.0" encoding="utf-8"?> | <?xml version="1.0" encoding="utf-8"?> | ||||||
| <manifest package="com.example"/> | <manifest package="com.hexated"/> | ||||||
							
								
								
									
										213
									
								
								DramaidProvider/src/main/kotlin/com/hexated/DramaidProvider.kt
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										213
									
								
								DramaidProvider/src/main/kotlin/com/hexated/DramaidProvider.kt
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,213 @@ | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.fasterxml.jackson.annotation.JsonProperty | ||||||
|  | import com.lagradost.cloudstream3.* | ||||||
|  | import com.lagradost.cloudstream3.utils.AppUtils.tryParseJson | ||||||
|  | import com.lagradost.cloudstream3.utils.ExtractorLink | ||||||
|  | import com.lagradost.cloudstream3.utils.getQualityFromName | ||||||
|  | import com.lagradost.cloudstream3.utils.loadExtractor | ||||||
|  | import org.jsoup.Jsoup | ||||||
|  | import org.jsoup.nodes.Element | ||||||
|  | 
 | ||||||
|  | class DramaidProvider : MainAPI() { | ||||||
|  |     override var mainUrl = "https://dramaid.asia" | ||||||
|  |     override var name = "DramaId" | ||||||
|  |     override val hasQuickSearch = false | ||||||
|  |     override val hasMainPage = true | ||||||
|  |     override var lang = "id" | ||||||
|  |     override val hasDownloadSupport = true | ||||||
|  |     override val hasChromecastSupport = false | ||||||
|  |     override val supportedTypes = setOf(TvType.AsianDrama) | ||||||
|  | 
 | ||||||
|  |     companion object { | ||||||
|  |         fun getStatus(t: String): ShowStatus { | ||||||
|  |             return when (t) { | ||||||
|  |                 "Completed" -> ShowStatus.Completed | ||||||
|  |                 "Ongoing" -> ShowStatus.Ongoing | ||||||
|  |                 else -> ShowStatus.Completed | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override val mainPage = mainPageOf( | ||||||
|  |         "&status=&type=&order=update" to "Drama Terbaru", | ||||||
|  |         "&order=latest" to "Baru Ditambahkan", | ||||||
|  |         "&status=&type=&order=popular" to "Drama Popular", | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     override suspend fun getMainPage(page: Int, request: MainPageRequest): HomePageResponse { | ||||||
|  |         val document = app.get("$mainUrl/series/?page=$page${request.data}").document | ||||||
|  |         val home = document.select("article[itemscope=itemscope]").mapNotNull { | ||||||
|  |             it.toSearchResult() | ||||||
|  |         } | ||||||
|  |         return newHomePageResponse(request.name, home) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun getProperDramaLink(uri: String): String { | ||||||
|  |         return if (uri.contains("/series/")) { | ||||||
|  |             uri | ||||||
|  |         } else { | ||||||
|  |             "$mainUrl/series/" + Regex("$mainUrl/(.+)-ep.+").find(uri)?.groupValues?.get(1) | ||||||
|  |                 .toString() | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun Element.toSearchResult(): SearchResponse? { | ||||||
|  |         val href = getProperDramaLink(this.selectFirst("a.tip")!!.attr("href")) | ||||||
|  |         val title = this.selectFirst("h2[itemprop=headline]")?.text()?.trim() ?: return null | ||||||
|  |         val posterUrl = fixUrlNull(this.selectFirst(".limit > noscript > img")?.attr("src")) | ||||||
|  | 
 | ||||||
|  |         return newTvSeriesSearchResponse(title, href, TvType.AsianDrama) { | ||||||
|  |             this.posterUrl = posterUrl | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun search(query: String): List<SearchResponse> { | ||||||
|  |         val link = "$mainUrl/?s=$query" | ||||||
|  |         val document = app.get(link).document | ||||||
|  | 
 | ||||||
|  |         return document.select("article[itemscope=itemscope]").map { | ||||||
|  |             val title = it.selectFirst("h2[itemprop=headline]")!!.text().trim() | ||||||
|  |             val poster = it.selectFirst(".limit > noscript > img")!!.attr("src") | ||||||
|  |             val href = it.selectFirst("a.tip")!!.attr("href") | ||||||
|  | 
 | ||||||
|  |             newTvSeriesSearchResponse(title, href, TvType.AsianDrama) { | ||||||
|  |                 this.posterUrl = poster | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun load(url: String): LoadResponse { | ||||||
|  |         val document = app.get(url).document | ||||||
|  | 
 | ||||||
|  |         val title = document.selectFirst("h1.entry-title")!!.text().trim() | ||||||
|  |         val poster = document.select(".thumb > noscript > img").attr("src") | ||||||
|  |         val tags = document.select(".genxed > a").map { it.text() } | ||||||
|  | 
 | ||||||
|  |         val year = Regex("\\d, ([0-9]*)").find( | ||||||
|  |             document.selectFirst(".info-content > .spe > span > time")!!.text().trim() | ||||||
|  |         )?.groupValues?.get(1).toString().toIntOrNull() | ||||||
|  |         val status = getStatus( | ||||||
|  |             document.select(".info-content > .spe > span:nth-child(1)") | ||||||
|  |                 .text().trim().replace("Status: ", "") | ||||||
|  |         ) | ||||||
|  |         val description = document.select(".entry-content > p").text().trim() | ||||||
|  | 
 | ||||||
|  |         val episodes = document.select(".eplister > ul > li").map { | ||||||
|  |             val name = it.selectFirst("a > .epl-title")!!.text().trim() | ||||||
|  |             val link = it.select("a").attr("href") | ||||||
|  |             val epNum = it.selectFirst("a > .epl-num")!!.text().trim().toIntOrNull() | ||||||
|  |             newEpisode(link) { | ||||||
|  |                 this.name = name | ||||||
|  |                 this.episode = epNum | ||||||
|  |             } | ||||||
|  |         }.reversed() | ||||||
|  | 
 | ||||||
|  |         val recommendations = | ||||||
|  |             document.select(".listupd > article[itemscope=itemscope]").map { rec -> | ||||||
|  |                 val epTitle = rec.selectFirst("h2[itemprop=headline]")!!.text().trim() | ||||||
|  |                 val epPoster = rec.selectFirst(".limit > noscript > img")!!.attr("src") | ||||||
|  |                 val epHref = fixUrl(rec.selectFirst("a.tip")!!.attr("href")) | ||||||
|  | 
 | ||||||
|  |                 newTvSeriesSearchResponse(epTitle, epHref, TvType.AsianDrama) { | ||||||
|  |                     this.posterUrl = epPoster | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |         if (episodes.size == 1) { | ||||||
|  |             return newMovieLoadResponse(title, url, TvType.Movie, episodes[0].data) { | ||||||
|  |                 posterUrl = poster | ||||||
|  |                 this.year = year | ||||||
|  |                 plot = description | ||||||
|  |                 this.tags = tags | ||||||
|  |                 this.recommendations = recommendations | ||||||
|  |             } | ||||||
|  |         } else { | ||||||
|  |             return newTvSeriesLoadResponse(title, url, TvType.AsianDrama, episodes = episodes) { | ||||||
|  |                 posterUrl = poster | ||||||
|  |                 this.year = year | ||||||
|  |                 showStatus = status | ||||||
|  |                 plot = description | ||||||
|  |                 this.tags = tags | ||||||
|  |                 this.recommendations = recommendations | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private data class Sources( | ||||||
|  |         @JsonProperty("file") val file: String, | ||||||
|  |         @JsonProperty("label") val label: String, | ||||||
|  |         @JsonProperty("type") val type: String, | ||||||
|  |         @JsonProperty("default") val default: Boolean? | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     private data class Tracks( | ||||||
|  |         @JsonProperty("file") val file: String, | ||||||
|  |         @JsonProperty("label") val label: String, | ||||||
|  |         @JsonProperty("kind") val type: String, | ||||||
|  |         @JsonProperty("default") val default: Boolean? | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     private suspend fun invokeDriveSource( | ||||||
|  |         url: String, | ||||||
|  |         name: String, | ||||||
|  |         subCallback: (SubtitleFile) -> Unit, | ||||||
|  |         sourceCallback: (ExtractorLink) -> Unit | ||||||
|  |     ) { | ||||||
|  |         val server = app.get(url).document.selectFirst(".picasa")?.nextElementSibling()?.data() | ||||||
|  | 
 | ||||||
|  |         val source = "[${server!!.substringAfter("sources: [").substringBefore("],")}]".trimIndent() | ||||||
|  |         val trackers = server.substringAfter("tracks:[").substringBefore("],") | ||||||
|  |             .replace("//language", "") | ||||||
|  |             .replace("file", "\"file\"") | ||||||
|  |             .replace("label", "\"label\"") | ||||||
|  |             .replace("kind", "\"kind\"").trimIndent() | ||||||
|  | 
 | ||||||
|  |         tryParseJson<List<Sources>>(source)?.map { | ||||||
|  |             sourceCallback( | ||||||
|  |                 ExtractorLink( | ||||||
|  |                     name, | ||||||
|  |                     "Drive", | ||||||
|  |                     fixUrl(it.file), | ||||||
|  |                     referer = "https://motonews.club/", | ||||||
|  |                     quality = getQualityFromName(it.label) | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         tryParseJson<Tracks>(trackers)?.let { | ||||||
|  |             subCallback.invoke( | ||||||
|  |                 SubtitleFile( | ||||||
|  |                     if (it.label.contains("Indonesia")) "${it.label}n" else it.label, | ||||||
|  |                     it.file | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun loadLinks( | ||||||
|  |         data: String, | ||||||
|  |         isCasting: Boolean, | ||||||
|  |         subtitleCallback: (SubtitleFile) -> Unit, | ||||||
|  |         callback: (ExtractorLink) -> Unit | ||||||
|  |     ): Boolean { | ||||||
|  |         val document = app.get(data).document | ||||||
|  |         val sources = document.select(".mobius > .mirror > option").mapNotNull { | ||||||
|  |             fixUrl(Jsoup.parse(base64Decode(it.attr("value"))).select("iframe").attr("src")) | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         sources.map { | ||||||
|  |             it.replace("https://ndrama.xyz", "https://www.fembed.com") | ||||||
|  |         }.apmap { | ||||||
|  |             when { | ||||||
|  |                 it.contains("motonews.club") -> invokeDriveSource(it, this.name, subtitleCallback, callback) | ||||||
|  |                 else -> loadExtractor(it, data, subtitleCallback, callback) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return true | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | @ -0,0 +1,14 @@ | ||||||
|  | 
 | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.lagradost.cloudstream3.plugins.CloudstreamPlugin | ||||||
|  | import com.lagradost.cloudstream3.plugins.Plugin | ||||||
|  | import android.content.Context | ||||||
|  | 
 | ||||||
|  | @CloudstreamPlugin | ||||||
|  | class DramaidProviderPlugin: Plugin() { | ||||||
|  |     override fun load(context: Context) { | ||||||
|  |         // All providers should be added in this manner. Please don't edit the providers list directly. | ||||||
|  |         registerMainAPI(DramaidProvider()) | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										27
									
								
								DubokuProvider/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								DubokuProvider/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,27 @@ | ||||||
|  | // use an integer for version numbers | ||||||
|  | version = 1 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | cloudstream { | ||||||
|  |     language = "zh" | ||||||
|  |     // All of these properties are optional, you can safely remove them | ||||||
|  | 
 | ||||||
|  |     // description = "Lorem Ipsum" | ||||||
|  |     authors = listOf("Hexated") | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Status int as the following: | ||||||
|  |      * 0: Down | ||||||
|  |      * 1: Ok | ||||||
|  |      * 2: Slow | ||||||
|  |      * 3: Beta only | ||||||
|  |      * */ | ||||||
|  |     status = 1 // will be 3 if unspecified | ||||||
|  |     tvTypes = listOf( | ||||||
|  |         "AsianDrama", | ||||||
|  |         "TvSeries", | ||||||
|  |         "Movie", | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     iconUrl = "https://www.google.com/s2/favicons?domain=www.duboku.tv&sz=%size%" | ||||||
|  | } | ||||||
							
								
								
									
										2
									
								
								DubokuProvider/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								DubokuProvider/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,2 @@ | ||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <manifest package="com.hexated"/> | ||||||
							
								
								
									
										133
									
								
								DubokuProvider/src/main/kotlin/com/hexated/DubokuProvider.kt
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										133
									
								
								DubokuProvider/src/main/kotlin/com/hexated/DubokuProvider.kt
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,133 @@ | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.fasterxml.jackson.annotation.JsonProperty | ||||||
|  | import com.lagradost.cloudstream3.* | ||||||
|  | import com.lagradost.cloudstream3.LoadResponse.Companion.addActors | ||||||
|  | import com.lagradost.cloudstream3.utils.AppUtils.tryParseJson | ||||||
|  | import com.lagradost.cloudstream3.utils.ExtractorLink | ||||||
|  | import com.lagradost.cloudstream3.utils.M3u8Helper | ||||||
|  | import org.jsoup.nodes.Element | ||||||
|  | 
 | ||||||
|  | class DubokuProvider : MainAPI() { | ||||||
|  |     override var mainUrl = "https://www.duboku.tv" | ||||||
|  |     override var name = "Duboku" | ||||||
|  |     override val hasMainPage = true | ||||||
|  |     override var lang = "zh" | ||||||
|  |     override val hasDownloadSupport = true | ||||||
|  |     override val supportedTypes = setOf( | ||||||
|  |         TvType.Movie, | ||||||
|  |         TvType.TvSeries, | ||||||
|  |         TvType.AsianDrama, | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     override val mainPage = mainPageOf( | ||||||
|  |         "$mainUrl/vodshow/2--time------" to "连续剧 时间", | ||||||
|  |         "$mainUrl/vodshow/2--hits------" to "连续剧 人气", | ||||||
|  |         "$mainUrl/vodshow/13--time------" to "陆剧 时间", | ||||||
|  |         "$mainUrl/vodshow/13--hits------" to "陆剧 人气", | ||||||
|  |         "$mainUrl/vodshow/15--time------" to "日韩剧 时间", | ||||||
|  |         "$mainUrl/vodshow/15--hits------" to "日韩剧 人气", | ||||||
|  |         "$mainUrl/vodshow/21--time------" to "短剧 时间", | ||||||
|  |         "$mainUrl/vodshow/21--hits------" to "短剧 人气", | ||||||
|  |         "$mainUrl/vodshow/16--time------" to "英美剧 时间", | ||||||
|  |         "$mainUrl/vodshow/16--hits------" to "英美剧 人气", | ||||||
|  |         "$mainUrl/vodshow/14--time------" to "台泰剧 时间", | ||||||
|  |         "$mainUrl/vodshow/14--hits------" to "台泰剧 人气", | ||||||
|  |         "$mainUrl/vodshow/20--time------" to "港剧 时间", | ||||||
|  |         "$mainUrl/vodshow/20--hits------" to "港剧 人气", | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     override suspend fun getMainPage( | ||||||
|  |         page: Int, | ||||||
|  |         request: MainPageRequest | ||||||
|  |     ): HomePageResponse { | ||||||
|  |         val document = app.get("${request.data}$page---.html").document | ||||||
|  |         val home = document.select("ul.myui-vodlist.clearfix li").mapNotNull { | ||||||
|  |             it.toSearchResult() | ||||||
|  |         } | ||||||
|  |         return newHomePageResponse(request.name, home) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun Element.toSearchResult(): SearchResponse? { | ||||||
|  |         val title = this.selectFirst("h4.title a")?.text()?.trim() ?: return null | ||||||
|  |         val href = fixUrl(this.selectFirst("a")?.attr("href").toString()) | ||||||
|  |         val posterUrl = fixUrlNull(this.selectFirst("a")?.attr("data-original")) | ||||||
|  |         val episode = this.selectFirst("span.pic-text.text-right")?.text()?.filter { it.isDigit() } | ||||||
|  |             ?.toIntOrNull() | ||||||
|  | 
 | ||||||
|  |         return newAnimeSearchResponse(title, href, TvType.Movie) { | ||||||
|  |             this.posterUrl = posterUrl | ||||||
|  |             addSub(episode) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun search(query: String): List<SearchResponse> { | ||||||
|  |         val document = app.get("$mainUrl/vodsearch/-------------.html?wd=$query&submit=").document | ||||||
|  | 
 | ||||||
|  |         return document.select("ul#searchList li").mapNotNull { | ||||||
|  |             it.toSearchResult() | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun load(url: String): LoadResponse? { | ||||||
|  |         val document = app.get(url).document | ||||||
|  | 
 | ||||||
|  |         val title = document.selectFirst("h1.title")?.text()?.trim() ?: return null | ||||||
|  |         val tvType = if (document.select("ul.myui-content__list li").size == 1 | ||||||
|  |         ) TvType.Movie else TvType.TvSeries | ||||||
|  |         val actors = document.select("p.data")[2].select("a").map { it.text() } | ||||||
|  | 
 | ||||||
|  |         val episodes = document.select("ul.myui-content__list li").map { | ||||||
|  |             val href = fixUrl(it.select("a").attr("href")) | ||||||
|  |             val name = it.select("a").text().trim() | ||||||
|  |             Episode( | ||||||
|  |                 data = href, | ||||||
|  |                 name = name, | ||||||
|  |             ) | ||||||
|  |         } | ||||||
|  |         return newTvSeriesLoadResponse(title, url, tvType, episodes) { | ||||||
|  |             this.posterUrl = fixUrlNull( | ||||||
|  |                 document.selectFirst("a.myui-vodlist__thumb.picture img")?.attr("data-original") | ||||||
|  |             ) | ||||||
|  |             this.year = | ||||||
|  |                 document.select("p.data")[0].select("a").last()?.text()?.trim()?.toIntOrNull() | ||||||
|  |             this.plot = document.selectFirst("span.sketch.content")?.text()?.trim() | ||||||
|  |             this.tags = document.select("p.data")[0].select("a").map { it.text() } | ||||||
|  |             this.rating = document.select("div#rating span.branch").text().toRatingInt() | ||||||
|  |             addActors(actors) | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun loadLinks( | ||||||
|  |         data: String, | ||||||
|  |         isCasting: Boolean, | ||||||
|  |         subtitleCallback: (SubtitleFile) -> Unit, | ||||||
|  |         callback: (ExtractorLink) -> Unit | ||||||
|  |     ): Boolean { | ||||||
|  | 
 | ||||||
|  |         app.get(data).document.select("script").map { script -> | ||||||
|  |             if (script.data().contains("var player_data={")) { | ||||||
|  |                 val dataJson = | ||||||
|  |                     script.data().substringAfter("var player_data={").substringBefore("}") | ||||||
|  |                 tryParseJson<Sources>("{$dataJson}")?.let { source -> | ||||||
|  |                     M3u8Helper.generateM3u8( | ||||||
|  |                         this.name, | ||||||
|  |                         source.url ?: return@map, | ||||||
|  |                         referer = "https://w.duboku.io/", | ||||||
|  |                         headers = mapOf("Origin" to "https://w.duboku.io") | ||||||
|  |                     ).forEach(callback) | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |         return true | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     data class Sources( | ||||||
|  |         @JsonProperty("url") val url: String?, | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | @ -0,0 +1,14 @@ | ||||||
|  | 
 | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.lagradost.cloudstream3.plugins.CloudstreamPlugin | ||||||
|  | import com.lagradost.cloudstream3.plugins.Plugin | ||||||
|  | import android.content.Context | ||||||
|  | 
 | ||||||
|  | @CloudstreamPlugin | ||||||
|  | class DubokuProviderPlugin: Plugin() { | ||||||
|  |     override fun load(context: Context) { | ||||||
|  |         // All providers should be added in this manner. Please don't edit the providers list directly. | ||||||
|  |         registerMainAPI(DubokuProvider()) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -1,24 +0,0 @@ | ||||||
| // 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 |  | ||||||
| 
 |  | ||||||
|     // List of video source types. Users are able to filter for extensions in a given category. |  | ||||||
|     // You can find a list of avaliable types here: |  | ||||||
|     // https://recloudstream.github.io/cloudstream/html/app/com.lagradost.cloudstream3/-tv-type/index.html |  | ||||||
|     tvTypes = listOf("Others") |  | ||||||
| } |  | ||||||
|  | @ -1,21 +0,0 @@ | ||||||
| package com.example |  | ||||||
| 
 |  | ||||||
| import com.lagradost.cloudstream3.TvType |  | ||||||
| import com.lagradost.cloudstream3.MainAPI |  | ||||||
| import com.lagradost.cloudstream3.SearchResponse |  | ||||||
| 
 |  | ||||||
| class ExampleProvider : MainAPI() { // all providers must be an instance of MainAPI |  | ||||||
|     override var mainUrl = "https://example.com/"  |  | ||||||
|     override var name = "Example provider" |  | ||||||
|     override val supportedTypes = setOf(TvType.Movie) |  | ||||||
| 
 |  | ||||||
|     override var lang = "en" |  | ||||||
| 
 |  | ||||||
|     // enable this when your provider has a main page |  | ||||||
|     override val hasMainPage = true |  | ||||||
| 
 |  | ||||||
|     // this function gets called when you search for something |  | ||||||
|     override suspend fun search(query: String): List<SearchResponse> { |  | ||||||
|         return listOf<SearchResponse>() |  | ||||||
|     } |  | ||||||
| } |  | ||||||
							
								
								
									
										27
									
								
								GomunimeProvider/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								GomunimeProvider/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,27 @@ | ||||||
|  | // use an integer for version numbers | ||||||
|  | version = 2 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | cloudstream { | ||||||
|  |     language = "id" | ||||||
|  |     // All of these properties are optional, you can safely remove them | ||||||
|  | 
 | ||||||
|  | //     description = "Lorem Ipsum" | ||||||
|  |      authors = listOf("Hexated") | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Status int as the following: | ||||||
|  |      * 0: Down | ||||||
|  |      * 1: Ok | ||||||
|  |      * 2: Slow | ||||||
|  |      * 3: Beta only | ||||||
|  |      * */ | ||||||
|  |     status = 0 // will be 3 if unspecified | ||||||
|  |     tvTypes = listOf( | ||||||
|  |         "AnimeMovie", | ||||||
|  |         "Anime", | ||||||
|  |         "OVA", | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     iconUrl = "https://www.google.com/s2/favicons?domain=185.231.223.76&sz=%size%" | ||||||
|  | } | ||||||
							
								
								
									
										2
									
								
								GomunimeProvider/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								GomunimeProvider/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,2 @@ | ||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <manifest package="com.hexated"/> | ||||||
							
								
								
									
										238
									
								
								GomunimeProvider/src/main/kotlin/com/hexated/GomunimeProvider.kt
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										238
									
								
								GomunimeProvider/src/main/kotlin/com/hexated/GomunimeProvider.kt
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,238 @@ | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.fasterxml.jackson.annotation.JsonProperty | ||||||
|  | import com.lagradost.cloudstream3.* | ||||||
|  | import com.lagradost.cloudstream3.LoadResponse.Companion.addTrailer | ||||||
|  | import com.lagradost.cloudstream3.mvvm.safeApiCall | ||||||
|  | import com.lagradost.cloudstream3.utils.AppUtils.parseJson | ||||||
|  | import com.lagradost.cloudstream3.utils.ExtractorLink | ||||||
|  | import com.lagradost.cloudstream3.utils.M3u8Helper | ||||||
|  | import com.lagradost.cloudstream3.utils.Qualities | ||||||
|  | import com.lagradost.cloudstream3.utils.loadExtractor | ||||||
|  | import org.jsoup.Jsoup | ||||||
|  | 
 | ||||||
|  | class GomunimeProvider : MainAPI() { | ||||||
|  |     override var mainUrl = "https://185.231.223.76" | ||||||
|  |     override var name = "Gomunime" | ||||||
|  |     override val hasMainPage = true | ||||||
|  |     override var lang = "id" | ||||||
|  |     override val hasDownloadSupport = true | ||||||
|  | 
 | ||||||
|  |     override val supportedTypes = setOf( | ||||||
|  |         TvType.Anime, | ||||||
|  |         TvType.AnimeMovie, | ||||||
|  |         TvType.OVA | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     companion object { | ||||||
|  | 
 | ||||||
|  |         private const val mainServer = "https://path.onicdn.xyz/app/rapi.php" | ||||||
|  | 
 | ||||||
|  |         fun getType(t: String): TvType { | ||||||
|  |             return if (t.contains("OVA") || t.contains("Special")) TvType.OVA | ||||||
|  |             else if (t.contains("Movie")) TvType.AnimeMovie | ||||||
|  |             else TvType.Anime | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         fun getStatus(t: String): ShowStatus { | ||||||
|  |             return when (t) { | ||||||
|  |                 "Completed" -> ShowStatus.Completed | ||||||
|  |                 "Ongoing" -> ShowStatus.Ongoing | ||||||
|  |                 else -> ShowStatus.Completed | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override val mainPage = mainPageOf( | ||||||
|  |         "e" to "Episode Baru", | ||||||
|  |         "c" to "Completed", | ||||||
|  |         "la" to "Live Action", | ||||||
|  |         "t" to "Trending" | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     override suspend fun getMainPage( | ||||||
|  |         page: Int, | ||||||
|  |         request: MainPageRequest | ||||||
|  |     ): HomePageResponse { | ||||||
|  |         val home = Jsoup.parse( | ||||||
|  |             (app.post( | ||||||
|  |                 url = "$mainUrl/wp-admin/admin-ajax.php/wp-admin/admin-ajax.php", | ||||||
|  |                 headers = mapOf("Referer" to mainUrl), | ||||||
|  |                 data = mapOf( | ||||||
|  |                     "action" to "home_ajax", | ||||||
|  |                     "fungsi" to request.data, | ||||||
|  |                     "pag" to "$page" | ||||||
|  |                 ) | ||||||
|  |             ).parsedSafe<Response>()?.html ?: throw ErrorLoadingException("Invalid Json reponse")) | ||||||
|  |         ).select("li").mapNotNull { | ||||||
|  |             val title = it.selectFirst("a.name")?.text()?.trim() ?: return@mapNotNull null | ||||||
|  |             val href = getProperAnimeLink(it.selectFirst("a")!!.attr("href")) | ||||||
|  |             val posterUrl = it.selectFirst("img")?.attr("src") | ||||||
|  |             val type = getType(it.selectFirst(".taglist > span")!!.text().trim()) | ||||||
|  |             val epNum = it.select(".tag.ep").text().replace(Regex("[^0-9]"), "").trim() | ||||||
|  |                 .toIntOrNull() | ||||||
|  |             newAnimeSearchResponse(title, href, type) { | ||||||
|  |                 this.posterUrl = posterUrl | ||||||
|  |                 addSub(epNum) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return newHomePageResponse(request.name, home) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun getProperAnimeLink(uri: String): String { | ||||||
|  |         return if (uri.contains("-episode")) { | ||||||
|  |             val href = | ||||||
|  |                 "$mainUrl/anime/" + Regex("\\w\\d/(.*)-episode.*").find(uri)?.groupValues?.get(1) | ||||||
|  |                     .toString() | ||||||
|  |             when { | ||||||
|  |                 href.contains("pokemon") -> href.replace(Regex("-[0-9]+"), "") | ||||||
|  |                 else -> href | ||||||
|  |             } | ||||||
|  |         } else { | ||||||
|  |             uri | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun search(query: String): List<SearchResponse> { | ||||||
|  |         val link = "$mainUrl/?s=$query" | ||||||
|  |         val document = app.get(link).document | ||||||
|  | 
 | ||||||
|  |         return document.select(".anime-list > li").map { | ||||||
|  |             val title = it.selectFirst("a.name")!!.text() | ||||||
|  |             val poster = it.selectFirst("img")!!.attr("src") | ||||||
|  |             val tvType = getType(it.selectFirst(".taglist > span")?.text().toString()) | ||||||
|  |             val href = fixUrl(it.selectFirst("a.name")!!.attr("href")) | ||||||
|  | 
 | ||||||
|  |             newAnimeSearchResponse(title, href, tvType) { | ||||||
|  |                 this.posterUrl = poster | ||||||
|  |                 addDubStatus(dubExist = false, subExist = true) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun load(url: String): LoadResponse { | ||||||
|  |         val document = app.get(url).document | ||||||
|  | 
 | ||||||
|  |         val title = document.selectFirst(".entry-title")?.text().toString() | ||||||
|  |         val poster = document.selectFirst(".thumbposter > img")?.attr("data-lazy-src") | ||||||
|  |         val tags = document.select(".genxed > a").map { it.text() } | ||||||
|  | 
 | ||||||
|  |         val year = Regex("\\d, ([0-9]*)").find( | ||||||
|  |             document.select("time[itemprop = datePublished]").text() | ||||||
|  |         )?.groupValues?.get(1)?.toIntOrNull() | ||||||
|  |         val status = getStatus(document.selectFirst(".spe > span")!!.ownText()) | ||||||
|  |         val description = document.select("div[itemprop = description] > p").text() | ||||||
|  |         val trailer = document.selectFirst("div.embed-responsive noscript iframe")?.attr("src") | ||||||
|  |         val episodes = parseJson<List<EpisodeElement>>( | ||||||
|  |             Regex("var episodelist = (\\[.*])").find( | ||||||
|  |                 document.select(".bixbox.bxcl.epcheck > script").toString().trim() | ||||||
|  |             )?.groupValues?.get(1).toString().replace(Regex("""\\"""), "").trim() | ||||||
|  |         ).map { | ||||||
|  |             val name = | ||||||
|  |                 Regex("(Episode\\s?[0-9]+)").find(it.epTitle.toString())?.groupValues?.getOrNull(0) | ||||||
|  |                     ?: it.epTitle | ||||||
|  |             val link = it.epLink | ||||||
|  |             Episode(link, name) | ||||||
|  |         }.reversed() | ||||||
|  | 
 | ||||||
|  |         return newAnimeLoadResponse(title, url, TvType.Anime) { | ||||||
|  |             engName = title | ||||||
|  |             posterUrl = poster | ||||||
|  |             this.year = year | ||||||
|  |             addEpisodes(DubStatus.Subbed, episodes) | ||||||
|  |             showStatus = status | ||||||
|  |             plot = description | ||||||
|  |             this.tags = tags | ||||||
|  |             addTrailer(trailer) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun loadLinks( | ||||||
|  |         data: String, | ||||||
|  |         isCasting: Boolean, | ||||||
|  |         subtitleCallback: (SubtitleFile) -> Unit, | ||||||
|  |         callback: (ExtractorLink) -> Unit | ||||||
|  |     ): Boolean { | ||||||
|  |         val document = app.get(data).document | ||||||
|  |         val scriptData = document.select("aside.sidebar > script").dataNodes().toString() | ||||||
|  |         val key = scriptData.substringAfter("var a_ray = '").substringBefore("';") | ||||||
|  |         val title = scriptData.substringAfter("var judul_postingan = \"").substringBefore("\";") | ||||||
|  |             .replace(" ", "+") | ||||||
|  |         val image = document.select("img#tempvid").last()?.attr("src").toString() | ||||||
|  | 
 | ||||||
|  |         val sources: List<Pair<String, String>> = app.post( | ||||||
|  |             url = mainServer, | ||||||
|  |             data = mapOf("data" to key, "gambar" to image, "judul" to title, "func" to "mirror"), | ||||||
|  |             referer = "$mainUrl/" | ||||||
|  |         ).document.select("div.gomunime-server-mirror").map { | ||||||
|  |             Pair( | ||||||
|  |                 it.attr("data-vhash"), | ||||||
|  |                 it.attr("data-type") | ||||||
|  |             ) | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         sources.apmap { | ||||||
|  |             safeApiCall { | ||||||
|  |                 when { | ||||||
|  |                     it.second.contains("frame") -> { | ||||||
|  |                         loadExtractor(it.first, mainUrl, subtitleCallback, callback) | ||||||
|  |                     } | ||||||
|  |                     // Skip for now | ||||||
|  | //                    it.second.contains("hls") -> { | ||||||
|  | //                        app.post( | ||||||
|  | //                            url = mainServer, | ||||||
|  | //                            data = mapOf("fid" to it.first, "func" to "hls") | ||||||
|  | //                        ).text.let { link -> | ||||||
|  | //                            M3u8Helper.generateM3u8( | ||||||
|  | //                                this.name, | ||||||
|  | //                                link, | ||||||
|  | //                                "$mainUrl/", | ||||||
|  | //                                headers = mapOf("Origin" to mainUrl) | ||||||
|  | //                            ).forEach(callback) | ||||||
|  | //                        } | ||||||
|  | //                    } | ||||||
|  |                     it.second.contains("mp4") -> { | ||||||
|  |                         app.post( | ||||||
|  |                             url = mainServer, | ||||||
|  |                             data = mapOf("data" to it.first, "func" to "blogs") | ||||||
|  |                         ).parsed<List<MobiSource>>().map { | ||||||
|  |                             callback.invoke( | ||||||
|  |                                 ExtractorLink( | ||||||
|  |                                     source = name, | ||||||
|  |                                     name = "Mobi SD", | ||||||
|  |                                     url = it.file, | ||||||
|  |                                     referer = "$mainUrl/", | ||||||
|  |                                     quality = Qualities.P360.value | ||||||
|  |                                 ) | ||||||
|  |                             ) | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  |                     else -> null | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return true | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private data class Response( | ||||||
|  |         @JsonProperty("status") val status: Boolean, | ||||||
|  |         @JsonProperty("html") val html: String | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     data class MobiSource( | ||||||
|  |         @JsonProperty("file") val file: String, | ||||||
|  |         @JsonProperty("label") val label: String, | ||||||
|  |         @JsonProperty("type") val type: String | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     private data class EpisodeElement( | ||||||
|  |         @JsonProperty("data-index") val dataIndex: Long?, | ||||||
|  |         @JsonProperty("ep-num") val epNum: String?, | ||||||
|  |         @JsonProperty("ep-title") val epTitle: String?, | ||||||
|  |         @JsonProperty("ep-link") val epLink: String, | ||||||
|  |         @JsonProperty("ep-date") val epDate: String? | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | @ -0,0 +1,14 @@ | ||||||
|  | 
 | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.lagradost.cloudstream3.plugins.CloudstreamPlugin | ||||||
|  | import com.lagradost.cloudstream3.plugins.Plugin | ||||||
|  | import android.content.Context | ||||||
|  | 
 | ||||||
|  | @CloudstreamPlugin | ||||||
|  | class GomunimeProviderPlugin: Plugin() { | ||||||
|  |     override fun load(context: Context) { | ||||||
|  |         // All providers should be added in this manner. Please don't edit the providers list directly. | ||||||
|  |         registerMainAPI(GomunimeProvider()) | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										27
									
								
								Gomunimeis/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								Gomunimeis/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,27 @@ | ||||||
|  | // use an integer for version numbers | ||||||
|  | version = 1 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | cloudstream { | ||||||
|  |     language = "id" | ||||||
|  |     // All of these properties are optional, you can safely remove them | ||||||
|  | 
 | ||||||
|  |     // description = "Lorem Ipsum" | ||||||
|  |      authors = listOf("Hexated") | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * 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=gomunime.is&sz=%size%" | ||||||
|  | } | ||||||
							
								
								
									
										2
									
								
								Gomunimeis/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								Gomunimeis/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,2 @@ | ||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <manifest package="com.hexated"/> | ||||||
							
								
								
									
										162
									
								
								Gomunimeis/src/main/kotlin/com/hexated/Gomunimeis.kt
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										162
									
								
								Gomunimeis/src/main/kotlin/com/hexated/Gomunimeis.kt
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,162 @@ | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.fasterxml.jackson.annotation.JsonProperty | ||||||
|  | import com.lagradost.cloudstream3.* | ||||||
|  | import com.lagradost.cloudstream3.utils.* | ||||||
|  | import java.util.ArrayList | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | class Gomunimeis : MainAPI() { | ||||||
|  |     override var mainUrl = "https://gomunime.is" | ||||||
|  |     override var name = "Gomunime.is" | ||||||
|  |     override val hasMainPage = true | ||||||
|  |     override var lang = "id" | ||||||
|  |     override val hasQuickSearch = true | ||||||
|  |     override val hasDownloadSupport = true | ||||||
|  | 
 | ||||||
|  |     override val supportedTypes = setOf( | ||||||
|  |         TvType.Anime, | ||||||
|  |         TvType.AnimeMovie, | ||||||
|  |         TvType.OVA | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     companion object { | ||||||
|  | 
 | ||||||
|  |         private const val mainImageUrl = "https://upload.anoboy.live" | ||||||
|  | 
 | ||||||
|  |         fun getType(t: String): TvType { | ||||||
|  |             return if (t.contains("OVA", true) || t.contains("Special", true)) TvType.OVA | ||||||
|  |             else if (t.contains("Movie", true)) TvType.AnimeMovie | ||||||
|  |             else TvType.Anime | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         fun getStatus(t: String): ShowStatus { | ||||||
|  |             return when (t) { | ||||||
|  |                 "Completed" -> ShowStatus.Completed | ||||||
|  |                 "Ongoing" -> ShowStatus.Ongoing | ||||||
|  |                 else -> ShowStatus.Completed | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override val mainPage = mainPageOf( | ||||||
|  |         "&limit=12&action=load_movie_last_update&status=Ongoing" to "Episode Baru", | ||||||
|  |         "&limit=15&action=load_movie_last_update&status=Completed" to "Completed", | ||||||
|  |         "&limit=15&action=load_movie_last_update&type=Live Action" to "Live Action", | ||||||
|  |         "&limit=15&action=load_movie_trending" to "Trending" | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     override suspend fun getMainPage( | ||||||
|  |         page: Int, | ||||||
|  |         request: MainPageRequest | ||||||
|  |     ): HomePageResponse { | ||||||
|  |         val home = app.get( | ||||||
|  |             "$mainUrl/my-ajax?page=$page${request.data}", | ||||||
|  |             headers = mapOf("X-Requested-With" to "XMLHttpRequest") | ||||||
|  |         ) | ||||||
|  |             .parsedSafe<Responses>()?.data | ||||||
|  |             ?.mapNotNull { media -> | ||||||
|  |                 media.toSearchResponse() | ||||||
|  |             } ?: throw ErrorLoadingException("Invalid Json reponse") | ||||||
|  |         return newHomePageResponse(request.name, home) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun Anime.toSearchResponse(): SearchResponse? { | ||||||
|  | 
 | ||||||
|  |         return newAnimeSearchResponse( | ||||||
|  |             postTitle ?: return null, | ||||||
|  |             "$mainUrl/anime/$postName.$salt", | ||||||
|  |             TvType.TvSeries, | ||||||
|  |         ) { | ||||||
|  |             this.posterUrl = "$mainImageUrl/$image" | ||||||
|  |             addSub(totalEpisode?.toIntOrNull()) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun quickSearch(query: String): List<SearchResponse> = search(query) | ||||||
|  | 
 | ||||||
|  |     override suspend fun search(query: String): List<SearchResponse> { | ||||||
|  |         return app.get( | ||||||
|  |             "$mainUrl/my-ajax?page=1&limit=10&action=load_search_movie&keyword=$query", | ||||||
|  |             referer = "$mainUrl/search/?keyword=$query", | ||||||
|  |             headers = mapOf("X-Requested-With" to "XMLHttpRequest") | ||||||
|  |         ).parsedSafe<Responses>()?.data | ||||||
|  |             ?.mapNotNull { media -> | ||||||
|  |                 media.toSearchResponse() | ||||||
|  |             } ?: throw ErrorLoadingException("Invalid Json reponse") | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun load(url: String): LoadResponse { | ||||||
|  |         val document = app.get(url).document | ||||||
|  | 
 | ||||||
|  |         val title = document.selectFirst(".entry-title")?.text().toString() | ||||||
|  |         val poster = document.selectFirst(".thumbposter > img")?.attr("src") | ||||||
|  |         val tags = document.select(".genxed > a").map { it.text() } | ||||||
|  |         val type = getType(document.selectFirst("div.info-content .spe span:last-child")?.ownText().toString()) | ||||||
|  |         val year = Regex("\\d, ([0-9]*)").find( | ||||||
|  |             document.selectFirst("div.info-content .spe span.split")?.ownText().toString() | ||||||
|  |         )?.groupValues?.get(1)?.toIntOrNull() | ||||||
|  |         val status = getStatus(document.selectFirst(".spe > span")!!.ownText()) | ||||||
|  |         val description = document.select("div[itemprop = description] > p").text() | ||||||
|  | 
 | ||||||
|  |         val episodes = document.select(".eplister > ul > li").map { | ||||||
|  |             val header = it.select(".epl-title").text() | ||||||
|  |             val name = Regex("(Episode\\s?[0-9]+)").find(header)?.groupValues?.getOrNull(0) ?: header | ||||||
|  |             val link = it.select("a").attr("href") | ||||||
|  |             Episode(link, name) | ||||||
|  |         }.reversed() | ||||||
|  | 
 | ||||||
|  |         return newAnimeLoadResponse(title, url, type) { | ||||||
|  |             engName = title | ||||||
|  |             posterUrl = poster | ||||||
|  |             this.year = year | ||||||
|  |             addEpisodes(DubStatus.Subbed, episodes) | ||||||
|  |             showStatus = status | ||||||
|  |             plot = description | ||||||
|  |             this.tags = tags | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun loadLinks( | ||||||
|  |         data: String, | ||||||
|  |         isCasting: Boolean, | ||||||
|  |         subtitleCallback: (SubtitleFile) -> Unit, | ||||||
|  |         callback: (ExtractorLink) -> Unit | ||||||
|  |     ): Boolean { | ||||||
|  | 
 | ||||||
|  |         val document = app.get(data).document | ||||||
|  | 
 | ||||||
|  |         document.select("div.player-container iframe").attr("src").substringAfter("html#").let { id -> | ||||||
|  |             app.get("https://gomunimes.com/stream?id=$id").parsedSafe<Sources>()?.server?.streamsb?.link?.let { link -> | ||||||
|  |                 loadExtractor(link, "https://vidgomunime.xyz/", subtitleCallback, callback) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return true | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     data class Streamsb( | ||||||
|  |         @JsonProperty("link") val link: String?, | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     data class Server( | ||||||
|  |         @JsonProperty("streamsb") val streamsb: Streamsb?, | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     data class Sources( | ||||||
|  |         @JsonProperty("server") val server: Server?, | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     data class Responses( | ||||||
|  |         @JsonProperty("data") val data: ArrayList<Anime>? = arrayListOf(), | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     data class Anime( | ||||||
|  |         @JsonProperty("post_title") val postTitle: String?, | ||||||
|  |         @JsonProperty("post_name") val postName: String?, | ||||||
|  |         @JsonProperty("image") val image: String?, | ||||||
|  |         @JsonProperty("total_episode") val totalEpisode: String?, | ||||||
|  |         @JsonProperty("salt") val salt: String?, | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  | } | ||||||
							
								
								
									
										14
									
								
								Gomunimeis/src/main/kotlin/com/hexated/GomunimeisPlugin.kt
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								Gomunimeis/src/main/kotlin/com/hexated/GomunimeisPlugin.kt
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,14 @@ | ||||||
|  | 
 | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.lagradost.cloudstream3.plugins.CloudstreamPlugin | ||||||
|  | import com.lagradost.cloudstream3.plugins.Plugin | ||||||
|  | import android.content.Context | ||||||
|  | 
 | ||||||
|  | @CloudstreamPlugin | ||||||
|  | class GomunimeisPlugin: Plugin() { | ||||||
|  |     override fun load(context: Context) { | ||||||
|  |         // All providers should be added in this manner. Please don't edit the providers list directly. | ||||||
|  |         registerMainAPI(Gomunimeis()) | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										28
									
								
								HDrezkaProvider/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								HDrezkaProvider/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,28 @@ | ||||||
|  | // use an integer for version numbers | ||||||
|  | version = 2 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | cloudstream { | ||||||
|  |     language = "ru" | ||||||
|  |     // All of these properties are optional, you can safely remove them | ||||||
|  | 
 | ||||||
|  |     // description = "Lorem Ipsum" | ||||||
|  |      authors = listOf("Hexated") | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Status int as the following: | ||||||
|  |      * 0: Down | ||||||
|  |      * 1: Ok | ||||||
|  |      * 2: Slow | ||||||
|  |      * 3: Beta only | ||||||
|  |      * */ | ||||||
|  |     status = 1 // will be 3 if unspecified | ||||||
|  |     tvTypes = listOf( | ||||||
|  |         "AsianDrama", | ||||||
|  |         "Anime", | ||||||
|  |         "TvSeries", | ||||||
|  |         "Movie", | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     iconUrl = "https://www.google.com/s2/favicons?domain=hdrezka19139.org&sz=%size%" | ||||||
|  | } | ||||||
							
								
								
									
										2
									
								
								HDrezkaProvider/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								HDrezkaProvider/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,2 @@ | ||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <manifest package="com.hexated"/> | ||||||
							
								
								
									
										407
									
								
								HDrezkaProvider/src/main/kotlin/com/hexated/HDrezkaProvider.kt
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										407
									
								
								HDrezkaProvider/src/main/kotlin/com/hexated/HDrezkaProvider.kt
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,407 @@ | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.fasterxml.jackson.annotation.JsonProperty | ||||||
|  | import com.lagradost.cloudstream3.* | ||||||
|  | import com.lagradost.cloudstream3.LoadResponse.Companion.addActors | ||||||
|  | import com.lagradost.cloudstream3.LoadResponse.Companion.addTrailer | ||||||
|  | import com.lagradost.cloudstream3.mvvm.suspendSafeApiCall | ||||||
|  | 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.Qualities | ||||||
|  | import com.lagradost.cloudstream3.utils.getQualityFromName | ||||||
|  | import org.jsoup.Jsoup | ||||||
|  | import org.jsoup.nodes.Element | ||||||
|  | import java.util.* | ||||||
|  | 
 | ||||||
|  | class HDrezkaProvider : MainAPI() { | ||||||
|  |     override var mainUrl = "https://rezka.ag" | ||||||
|  |     override var name = "HDrezka" | ||||||
|  |     override val hasMainPage = true | ||||||
|  |     override var lang = "ru" | ||||||
|  |     override val hasDownloadSupport = true | ||||||
|  |     override val supportedTypes = setOf( | ||||||
|  |         TvType.Movie, | ||||||
|  |         TvType.TvSeries, | ||||||
|  |         TvType.Anime, | ||||||
|  |         TvType.AsianDrama | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     override val mainPage = mainPageOf( | ||||||
|  |         "$mainUrl/films/?filter=watching" to "фильмы", | ||||||
|  |         "$mainUrl/series/?filter=watching" to "сериалы", | ||||||
|  |         "$mainUrl/cartoons/?filter=watching" to "мультфильмы", | ||||||
|  |         "$mainUrl/animation/?filter=watching" to "аниме", | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     override suspend fun getMainPage( | ||||||
|  |         page: Int, | ||||||
|  |         request: MainPageRequest | ||||||
|  |     ): HomePageResponse { | ||||||
|  |         val url = request.data.split("?") | ||||||
|  |         val home = app.get("${url.first()}page/$page/?${url.last()}").document.select( | ||||||
|  |             "div.b-content__inline_items div.b-content__inline_item" | ||||||
|  |         ).map { | ||||||
|  |             it.toSearchResult() | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return newHomePageResponse(request.name, home) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun Element.toSearchResult(): SearchResponse { | ||||||
|  |         val title = | ||||||
|  |             this.selectFirst("div.b-content__inline_item-link > a")?.text()?.trim().toString() | ||||||
|  |         val href = this.selectFirst("a")?.attr("href").toString() | ||||||
|  |         val posterUrl = this.select("img").attr("src") | ||||||
|  |         val type = if (this.select("span.info").isNotEmpty()) TvType.TvSeries else TvType.Movie | ||||||
|  |         return if (type == TvType.Movie) { | ||||||
|  |             newMovieSearchResponse(title, href, TvType.Movie) { | ||||||
|  |                 this.posterUrl = posterUrl | ||||||
|  |             } | ||||||
|  |         } else { | ||||||
|  |             val episode = | ||||||
|  |                 this.select("span.info").text().substringAfter(",").replace(Regex("[^0-9]"), "") | ||||||
|  |                     .toIntOrNull() | ||||||
|  |             newAnimeSearchResponse(title, href, TvType.TvSeries) { | ||||||
|  |                 this.posterUrl = posterUrl | ||||||
|  |                 addDubStatus( | ||||||
|  |                     dubExist = true, | ||||||
|  |                     dubEpisodes = episode, | ||||||
|  |                     subExist = true, | ||||||
|  |                     subEpisodes = episode | ||||||
|  |                 ) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun search(query: String): List<SearchResponse> { | ||||||
|  |         val link = "$mainUrl/search/?do=search&subaction=search&q=$query" | ||||||
|  |         val document = app.get(link).document | ||||||
|  | 
 | ||||||
|  |         return document.select("div.b-content__inline_items div.b-content__inline_item").map { | ||||||
|  |             it.toSearchResult() | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun load(url: String): LoadResponse { | ||||||
|  |         val document = app.get(url).document | ||||||
|  | 
 | ||||||
|  |         val id = url.split("/").last().split("-").first() | ||||||
|  |         val title = (document.selectFirst("div.b-post__origtitle")?.text()?.trim() | ||||||
|  |             ?: document.selectFirst("div.b-post__title h1")?.text()?.trim()).toString() | ||||||
|  |         val poster = fixUrlNull(document.selectFirst("div.b-sidecover img")?.attr("src")) | ||||||
|  |         val tags = | ||||||
|  |             document.select("table.b-post__info > tbody > tr:nth-child(5) span[itemprop=genre]") | ||||||
|  |                 .map { it.text() } | ||||||
|  |         val year = document.select("div.film-info > div:nth-child(2) a").text().toIntOrNull() | ||||||
|  |         val tvType = if (document.select("div#simple-episodes-tabs") | ||||||
|  |                 .isNullOrEmpty() | ||||||
|  |         ) TvType.Movie else TvType.TvSeries | ||||||
|  |         val description = document.selectFirst("div.b-post__description_text")?.text()?.trim() | ||||||
|  |         val trailer = app.post( | ||||||
|  |             "$mainUrl/engine/ajax/gettrailervideo.php", | ||||||
|  |             data = mapOf("id" to id), | ||||||
|  |             referer = url | ||||||
|  |         ).parsedSafe<Trailer>()?.code.let { | ||||||
|  |             Jsoup.parse(it.toString()).select("iframe").attr("src") | ||||||
|  |         } | ||||||
|  |         val rating = | ||||||
|  |             document.selectFirst("table.b-post__info > tbody > tr:nth-child(1) span.bold")?.text() | ||||||
|  |                 .toRatingInt() | ||||||
|  |         val actors = document.select("table.b-post__info > tbody > tr:last-child span.item").map { | ||||||
|  |             Actor( | ||||||
|  |                 it.selectFirst("span[itemprop=name]")?.text().toString(), | ||||||
|  |                 it.selectFirst("span[itemprop=actor]")?.attr("data-photo") | ||||||
|  |             ) | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         val recommendations = document.select("div.b-sidelist div.b-content__inline_item").map { | ||||||
|  |             it.toSearchResult() | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         val data = HashMap<String, Any>() | ||||||
|  |         val server = ArrayList<Map<String, String>>() | ||||||
|  | 
 | ||||||
|  |         data["id"] = id | ||||||
|  |         data["favs"] = document.selectFirst("input#ctrl_favs")?.attr("value").toString() | ||||||
|  |         data["ref"] = url | ||||||
|  | 
 | ||||||
|  |         return if (tvType == TvType.TvSeries) { | ||||||
|  |             document.select("ul#translators-list li").map { res -> | ||||||
|  |                 server.add( | ||||||
|  |                     mapOf( | ||||||
|  |                         "translator_name" to res.text(), | ||||||
|  |                         "translator_id" to res.attr("data-translator_id"), | ||||||
|  |                     ) | ||||||
|  |                 ) | ||||||
|  |             } | ||||||
|  |             val episodes = document.select("div#simple-episodes-tabs ul li").map { | ||||||
|  |                 val season = it.attr("data-season_id").toIntOrNull() | ||||||
|  |                 val episode = it.attr("data-episode_id").toIntOrNull() | ||||||
|  |                 val name = "Episode $episode" | ||||||
|  | 
 | ||||||
|  |                 data["season"] = "$season" | ||||||
|  |                 data["episode"] = "$episode" | ||||||
|  |                 data["server"] = server | ||||||
|  |                 data["action"] = "get_stream" | ||||||
|  | 
 | ||||||
|  |                 Episode( | ||||||
|  |                     data.toJson(), | ||||||
|  |                     name, | ||||||
|  |                     season, | ||||||
|  |                     episode, | ||||||
|  |                 ) | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             newTvSeriesLoadResponse(title, url, TvType.TvSeries, episodes) { | ||||||
|  |                 this.posterUrl = poster | ||||||
|  |                 this.year = year | ||||||
|  |                 this.plot = description | ||||||
|  |                 this.tags = tags | ||||||
|  |                 this.rating = rating | ||||||
|  |                 addActors(actors) | ||||||
|  |                 this.recommendations = recommendations | ||||||
|  |                 addTrailer(trailer) | ||||||
|  |             } | ||||||
|  |         } else { | ||||||
|  |             document.select("ul#translators-list li").map { res -> | ||||||
|  |                 server.add( | ||||||
|  |                     mapOf( | ||||||
|  |                         "translator_name" to res.text(), | ||||||
|  |                         "translator_id" to res.attr("data-translator_id"), | ||||||
|  |                         "camrip" to res.attr("data-camrip"), | ||||||
|  |                         "ads" to res.attr("data-ads"), | ||||||
|  |                         "director" to res.attr("data-director") | ||||||
|  |                     ) | ||||||
|  |                 ) | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             data["server"] = server | ||||||
|  |             data["action"] = "get_movie" | ||||||
|  | 
 | ||||||
|  |             newMovieLoadResponse(title, url, TvType.Movie, data.toJson()) { | ||||||
|  |                 this.posterUrl = poster | ||||||
|  |                 this.year = year | ||||||
|  |                 this.plot = description | ||||||
|  |                 this.tags = tags | ||||||
|  |                 this.rating = rating | ||||||
|  |                 addActors(actors) | ||||||
|  |                 this.recommendations = recommendations | ||||||
|  |                 addTrailer(trailer) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun decryptStreamUrl(data: String): String { | ||||||
|  | 
 | ||||||
|  |         fun getTrash(arr: List<String>, item: Int): List<String> { | ||||||
|  |             val trash = ArrayList<List<String>>() | ||||||
|  |             for (i in 1..item) { | ||||||
|  |                 trash.add(arr) | ||||||
|  |             } | ||||||
|  |             return trash.reduce { acc, list -> | ||||||
|  |                 val temp = ArrayList<String>() | ||||||
|  |                 acc.forEach { ac -> | ||||||
|  |                     list.forEach { li -> | ||||||
|  |                         temp.add(ac.plus(li)) | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |                 return@reduce temp | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         val trashList = listOf("@", "#", "!", "^", "$") | ||||||
|  |         val trashSet = getTrash(trashList, 2) + getTrash(trashList, 3) | ||||||
|  |         var trashString = data.replace("#h", "").split("//_//").joinToString("") | ||||||
|  | 
 | ||||||
|  |         trashSet.forEach { | ||||||
|  |             val temp = base64Encode(it.toByteArray()) | ||||||
|  |             trashString = trashString.replace(temp, "") | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return base64Decode(trashString) | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun cleanCallback( | ||||||
|  |         source: String, | ||||||
|  |         url: String, | ||||||
|  |         quality: String, | ||||||
|  |         isM3u8: Boolean, | ||||||
|  |         sourceCallback: (ExtractorLink) -> Unit | ||||||
|  |     ) { | ||||||
|  |         sourceCallback.invoke( | ||||||
|  |             ExtractorLink( | ||||||
|  |                 source, | ||||||
|  |                 source, | ||||||
|  |                 url, | ||||||
|  |                 "$mainUrl/", | ||||||
|  |                 getQuality(quality), | ||||||
|  |                 isM3u8, | ||||||
|  |                 headers = mapOf( | ||||||
|  |                     "Origin" to mainUrl | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun getLanguage(str: String): String { | ||||||
|  |         return when (str) { | ||||||
|  |             "Русский" -> "Russian" | ||||||
|  |             "Українська" -> "Ukrainian" | ||||||
|  |             else -> str | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun getQuality(str: String): Int { | ||||||
|  |         return when (str) { | ||||||
|  |             "360p" -> Qualities.P240.value | ||||||
|  |             "480p" -> Qualities.P360.value | ||||||
|  |             "720p" -> Qualities.P480.value | ||||||
|  |             "1080p" -> Qualities.P720.value | ||||||
|  |             "1080p Ultra" -> Qualities.P1080.value | ||||||
|  |             else -> getQualityFromName(str) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun invokeSources( | ||||||
|  |         source: String, | ||||||
|  |         url: String, | ||||||
|  |         subtitle: String, | ||||||
|  |         subCallback: (SubtitleFile) -> Unit, | ||||||
|  |         sourceCallback: (ExtractorLink) -> Unit | ||||||
|  |     ) { | ||||||
|  |         decryptStreamUrl(url).split(",").map { links -> | ||||||
|  |             val quality = | ||||||
|  |                 Regex("\\[([0-9]*p.*?)]").find(links)?.groupValues?.getOrNull(1) | ||||||
|  |                     .toString().trim() | ||||||
|  |             links.replace("[$quality]", "").split("or").map { it.trim() } | ||||||
|  |                 .map { link -> | ||||||
|  | 
 | ||||||
|  |                     if (link.endsWith(".m3u8")) { | ||||||
|  |                         cleanCallback( | ||||||
|  |                             "$source (Main)", | ||||||
|  |                             link, | ||||||
|  |                             quality, | ||||||
|  |                             true, | ||||||
|  |                             sourceCallback, | ||||||
|  |                         ) | ||||||
|  |                     } else { | ||||||
|  |                         cleanCallback( | ||||||
|  |                             "$source (Backup)", | ||||||
|  |                             link, | ||||||
|  |                             quality, | ||||||
|  |                             false, | ||||||
|  |                             sourceCallback, | ||||||
|  |                         ) | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         subtitle.split(",").map { sub -> | ||||||
|  |             val language = | ||||||
|  |                 Regex("\\[(.*)]").find(sub)?.groupValues?.getOrNull(1) | ||||||
|  |                     .toString() | ||||||
|  |             val link = sub.replace("[$language]", "").trim() | ||||||
|  |             subCallback.invoke( | ||||||
|  |                 SubtitleFile( | ||||||
|  |                     getLanguage(language), | ||||||
|  |                     link | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun loadLinks( | ||||||
|  |         data: String, | ||||||
|  |         isCasting: Boolean, | ||||||
|  |         subtitleCallback: (SubtitleFile) -> Unit, | ||||||
|  |         callback: (ExtractorLink) -> Unit | ||||||
|  |     ): Boolean { | ||||||
|  | 
 | ||||||
|  |         tryParseJson<Data>(data)?.let { res -> | ||||||
|  |             if (res.server?.isEmpty() == true) { | ||||||
|  |                 val document = app.get(res.ref ?: return@let).document | ||||||
|  |                 document.select("script").map { script -> | ||||||
|  |                     if (script.data().contains("sof.tv.initCDNMoviesEvents(")) { | ||||||
|  |                         val dataJson = | ||||||
|  |                             script.data().substringAfter("false, {").substringBefore("});") | ||||||
|  |                         tryParseJson<LocalSources>("{$dataJson}")?.let { source -> | ||||||
|  |                             invokeSources( | ||||||
|  |                                 this.name, | ||||||
|  |                                 source.streams, | ||||||
|  |                                 source.subtitle.toString(), | ||||||
|  |                                 subtitleCallback, | ||||||
|  |                                 callback | ||||||
|  |                             ) | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } else { | ||||||
|  |                 res.server?.apmap { server -> | ||||||
|  |                     suspendSafeApiCall { | ||||||
|  |                         app.post( | ||||||
|  |                             url = "$mainUrl/ajax/get_cdn_series/?t=${Date().time}", | ||||||
|  |                             data = mapOf( | ||||||
|  |                                 "id" to res.id, | ||||||
|  |                                 "translator_id" to server.translator_id, | ||||||
|  |                                 "favs" to res.favs, | ||||||
|  |                                 "is_camrip" to server.camrip, | ||||||
|  |                                 "is_ads" to server.ads, | ||||||
|  |                                 "is_director" to server.director, | ||||||
|  |                                 "season" to res.season, | ||||||
|  |                                 "episode" to res.episode, | ||||||
|  |                                 "action" to res.action, | ||||||
|  |                             ).filterValues { it != null }.mapValues { it.value as String }, | ||||||
|  |                             referer = res.ref | ||||||
|  |                         ).parsedSafe<Sources>()?.let { source -> | ||||||
|  |                             invokeSources( | ||||||
|  |                                 server.translator_name.toString(), | ||||||
|  |                                 source.url, | ||||||
|  |                                 source.subtitle.toString(), | ||||||
|  |                                 subtitleCallback, | ||||||
|  |                                 callback | ||||||
|  |                             ) | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return true | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     data class LocalSources( | ||||||
|  |         @JsonProperty("streams") val streams: String, | ||||||
|  |         @JsonProperty("subtitle") val subtitle: Any?, | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     data class Sources( | ||||||
|  |         @JsonProperty("url") val url: String, | ||||||
|  |         @JsonProperty("subtitle") val subtitle: Any?, | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     data class Server( | ||||||
|  |         @JsonProperty("translator_name") val translator_name: String?, | ||||||
|  |         @JsonProperty("translator_id") val translator_id: String?, | ||||||
|  |         @JsonProperty("camrip") val camrip: String?, | ||||||
|  |         @JsonProperty("ads") val ads: String?, | ||||||
|  |         @JsonProperty("director") val director: String?, | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     data class Data( | ||||||
|  |         @JsonProperty("id") val id: String?, | ||||||
|  |         @JsonProperty("favs") val favs: String?, | ||||||
|  |         @JsonProperty("server") val server: List<Server>?, | ||||||
|  |         @JsonProperty("season") val season: String?, | ||||||
|  |         @JsonProperty("episode") val episode: String?, | ||||||
|  |         @JsonProperty("action") val action: String?, | ||||||
|  |         @JsonProperty("ref") val ref: String?, | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     data class Trailer( | ||||||
|  |         @JsonProperty("success") val success: Boolean?, | ||||||
|  |         @JsonProperty("code") val code: String?, | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | @ -0,0 +1,14 @@ | ||||||
|  | 
 | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.lagradost.cloudstream3.plugins.CloudstreamPlugin | ||||||
|  | import com.lagradost.cloudstream3.plugins.Plugin | ||||||
|  | import android.content.Context | ||||||
|  | 
 | ||||||
|  | @CloudstreamPlugin | ||||||
|  | class HDrezkaProviderPlugin: Plugin() { | ||||||
|  |     override fun load(context: Context) { | ||||||
|  |         // All providers should be added in this manner. Please don't edit the providers list directly. | ||||||
|  |         registerMainAPI(HDrezkaProvider()) | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										26
									
								
								Hdfilmcehennemi/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								Hdfilmcehennemi/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,26 @@ | ||||||
|  | // use an integer for version numbers | ||||||
|  | version = 1 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | cloudstream { | ||||||
|  |     language = "tr" | ||||||
|  |     // All of these properties are optional, you can safely remove them | ||||||
|  | 
 | ||||||
|  |     // description = "Lorem Ipsum" | ||||||
|  |      authors = listOf("Hexated") | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * 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=hdfilmcehennemi.live&sz=%size%" | ||||||
|  | } | ||||||
							
								
								
									
										2
									
								
								Hdfilmcehennemi/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								Hdfilmcehennemi/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,2 @@ | ||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <manifest package="com.hexated"/> | ||||||
							
								
								
									
										205
									
								
								Hdfilmcehennemi/src/main/kotlin/com/hexated/Hdfilmcehennemi.kt
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										205
									
								
								Hdfilmcehennemi/src/main/kotlin/com/hexated/Hdfilmcehennemi.kt
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,205 @@ | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.fasterxml.jackson.annotation.JsonProperty | ||||||
|  | import com.lagradost.cloudstream3.* | ||||||
|  | import com.lagradost.cloudstream3.LoadResponse.Companion.addActors | ||||||
|  | import com.lagradost.cloudstream3.LoadResponse.Companion.addTrailer | ||||||
|  | import com.lagradost.cloudstream3.mvvm.safeApiCall | ||||||
|  | import com.lagradost.cloudstream3.utils.* | ||||||
|  | import org.jsoup.nodes.Element | ||||||
|  | 
 | ||||||
|  | class Hdfilmcehennemi : MainAPI() { | ||||||
|  |     override var mainUrl = "https://www.hdfilmcehennemi.live" | ||||||
|  |     override var name = "hdfilmcehennemi" | ||||||
|  |     override val hasMainPage = true | ||||||
|  |     override var lang = "tr" | ||||||
|  |     override val hasQuickSearch = true | ||||||
|  |     override val hasDownloadSupport = true | ||||||
|  |     override val supportedTypes = setOf( | ||||||
|  |         TvType.Movie, | ||||||
|  |         TvType.TvSeries, | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     override val mainPage = mainPageOf( | ||||||
|  |         "$mainUrl/category/tavsiye-filmler-izle1/page/" to "Tavsiye Filmler Kategorisi", | ||||||
|  |         "$mainUrl/yabancidizi/page/" to "Son Eklenen Yabancı Diziler", | ||||||
|  |         "$mainUrl/imdb-7-puan-uzeri-filmler/page/" to "Imdb 7+ Filmler", | ||||||
|  |         "$mainUrl/en-cok-yorumlananlar/page/" to "En Çok Yorumlananlar", | ||||||
|  |         "$mainUrl/en-cok-begenilen-filmleri-izle/page/" to "En Çok Beğenilenler", | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     override suspend fun getMainPage( | ||||||
|  |         page: Int, | ||||||
|  |         request: MainPageRequest | ||||||
|  |     ): HomePageResponse { | ||||||
|  |         val document = app.get(request.data + page).document | ||||||
|  |         val home = document.select("div.card-body div.row div.col-6.col-sm-3.poster-container") | ||||||
|  |             .mapNotNull { | ||||||
|  |                 it.toSearchResult() | ||||||
|  |             } | ||||||
|  |         return newHomePageResponse(request.name, home) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun Element.toSearchResult(): SearchResponse? { | ||||||
|  |         val title = this.selectFirst("a")?.text() ?: return null | ||||||
|  |         val href = fixUrlNull(this.selectFirst("a")?.attr("href")) ?: return null | ||||||
|  |         val posterUrl = fixUrlNull(this.selectFirst("img")?.attr("data-src")) | ||||||
|  |         return newMovieSearchResponse(title, href, TvType.Movie) { | ||||||
|  |             this.posterUrl = posterUrl | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun Media.toSearchResponse(): SearchResponse? { | ||||||
|  |         return newMovieSearchResponse( | ||||||
|  |             title ?: return null, | ||||||
|  |             "$mainUrl/$slugPrefix$slug", | ||||||
|  |             TvType.TvSeries, | ||||||
|  |         ) { | ||||||
|  |             this.posterUrl = "$mainUrl/uploads/poster/$poster" | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun quickSearch(query: String): List<SearchResponse> = search(query) | ||||||
|  | 
 | ||||||
|  |     override suspend fun search(query: String): List<SearchResponse> { | ||||||
|  |         return app.post( | ||||||
|  |             "$mainUrl/search/", | ||||||
|  |             data = mapOf("query" to query), | ||||||
|  |             referer = "$mainUrl/", | ||||||
|  |             headers = mapOf( | ||||||
|  |                 "Accept" to "application/json, text/javascript, */*; q=0.01", | ||||||
|  |                 "X-Requested-With" to "XMLHttpRequest" | ||||||
|  |             ) | ||||||
|  |         ).parsedSafe<Result>()?.result?.mapNotNull { media -> | ||||||
|  |             media.toSearchResponse() | ||||||
|  |         } ?: throw ErrorLoadingException("Invalid Json reponse") | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun load(url: String): LoadResponse? { | ||||||
|  |         val document = app.get(url).document | ||||||
|  | 
 | ||||||
|  |         val title = document.selectFirst("div.card-header > h1, div.card-header > h2")?.text() | ||||||
|  |             ?: return null | ||||||
|  |         val poster = fixUrlNull(document.selectFirst("img.img-fluid")?.attr("src")) | ||||||
|  |         val tags = document.select("div.mb-0.lh-lg div:nth-child(5) a").map { it.text() } | ||||||
|  |         val year = | ||||||
|  |             document.selectFirst("div.mb-0.lh-lg div:nth-child(4) a")?.text()?.trim()?.toIntOrNull() | ||||||
|  |         val tvType = if (document.select("nav#seasonsTabs").isNullOrEmpty() | ||||||
|  |         ) TvType.Movie else TvType.TvSeries | ||||||
|  |         val description = document.selectFirst("article.text-white > p")?.text()?.trim() | ||||||
|  |         val rating = document.selectFirst("div.rating-votes div.rate span")?.text()?.toRatingInt() | ||||||
|  |         val actors = document.select("div.mb-0.lh-lg div:last-child a.chip").map { | ||||||
|  |             Actor(it.text(), it.select("img").attr("src")) | ||||||
|  |         } | ||||||
|  |         val recommendations = | ||||||
|  |             document.select("div.swiper-wrapper div.poster.poster-pop").mapNotNull { | ||||||
|  |                 val recName = it.selectFirst("h2.title")?.text() ?: return@mapNotNull null | ||||||
|  |                 val recHref = | ||||||
|  |                     fixUrlNull(it.selectFirst("a")?.attr("href")) ?: return@mapNotNull null | ||||||
|  |                 val recPosterUrl = fixUrlNull(it.selectFirst("img")?.attr("data-src")) | ||||||
|  |                 newTvSeriesSearchResponse(recName, recHref, TvType.TvSeries) { | ||||||
|  |                     this.posterUrl = recPosterUrl | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |         return if (tvType == TvType.TvSeries) { | ||||||
|  |             val trailer = | ||||||
|  |                 document.selectFirst("button.btn.btn-fragman.btn-danger")?.attr("data-trailer") | ||||||
|  |                     ?.let { | ||||||
|  |                         "https://www.youtube.com/embed/$it" | ||||||
|  |                     } | ||||||
|  |             val episodes = document.select("div#seasonsTabs-tabContent div.card-list-item").map { | ||||||
|  |                 val href = it.select("a").attr("href") | ||||||
|  |                 val name = it.select("h3").text().trim() | ||||||
|  |                 val episode = it.select("h3").text().let { num -> | ||||||
|  |                     Regex("Sezon\\s?([0-9]+).").find(num)?.groupValues?.getOrNull(1)?.toIntOrNull() | ||||||
|  |                 } | ||||||
|  |                 val season = it.parents()[1].attr("id").substringAfter("-").toIntOrNull() | ||||||
|  |                 Episode( | ||||||
|  |                     href, | ||||||
|  |                     name, | ||||||
|  |                     season, | ||||||
|  |                     episode, | ||||||
|  |                 ) | ||||||
|  |             } | ||||||
|  |             newTvSeriesLoadResponse(title, url, TvType.TvSeries, episodes) { | ||||||
|  |                 this.posterUrl = poster | ||||||
|  |                 this.year = year | ||||||
|  |                 this.plot = description | ||||||
|  |                 this.tags = tags | ||||||
|  |                 this.rating = rating | ||||||
|  |                 addActors(actors) | ||||||
|  |                 this.recommendations = recommendations | ||||||
|  |                 addTrailer(trailer) | ||||||
|  |             } | ||||||
|  |         } else { | ||||||
|  |             val trailer = | ||||||
|  |                 document.selectFirst("nav.nav.card-nav.nav-slider a[data-bs-toggle=\"modal\"]") | ||||||
|  |                     ?.attr("data-trailer")?.let { | ||||||
|  |                         "https://www.youtube.com/embed/$it" | ||||||
|  |                     } | ||||||
|  |             newMovieLoadResponse(title, url, TvType.Movie, url) { | ||||||
|  |                 this.posterUrl = poster | ||||||
|  |                 this.year = year | ||||||
|  |                 this.plot = description | ||||||
|  |                 this.tags = tags | ||||||
|  |                 this.rating = rating | ||||||
|  |                 addActors(actors) | ||||||
|  |                 this.recommendations = recommendations | ||||||
|  |                 addTrailer(trailer) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private suspend fun invokeLocalSource( | ||||||
|  |         source: String, | ||||||
|  |         url: String, | ||||||
|  |         sourceCallback: (ExtractorLink) -> Unit | ||||||
|  |     ) { | ||||||
|  |         val m3uLink = | ||||||
|  |             app.get(url, referer = "$mainUrl/").document.select("script") | ||||||
|  |                 .find { | ||||||
|  |                     it.data().contains("var sources = [];") || it.data() | ||||||
|  |                         .contains("playerInstance =") | ||||||
|  |                 }?.data() | ||||||
|  |                 ?.substringAfter("[{file:\"")?.substringBefore("\"}]") ?: return | ||||||
|  | 
 | ||||||
|  |         M3u8Helper.generateM3u8( | ||||||
|  |             source, | ||||||
|  |             m3uLink, | ||||||
|  |             if (url.startsWith(mainUrl)) "$mainUrl/" else "https://vidmoly.to/" | ||||||
|  |         ).forEach(sourceCallback) | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun loadLinks( | ||||||
|  |         data: String, | ||||||
|  |         isCasting: Boolean, | ||||||
|  |         subtitleCallback: (SubtitleFile) -> Unit, | ||||||
|  |         callback: (ExtractorLink) -> Unit | ||||||
|  |     ): Boolean { | ||||||
|  |         app.get(data).document.select("nav.nav.card-nav.nav-slider a.nav-link").map { | ||||||
|  |             Pair(it.attr("href"), it.text()) | ||||||
|  |         }.apmap { (url, source) -> | ||||||
|  |             safeApiCall { | ||||||
|  |                 app.get(url).document.select("div.card-video > iframe").attr("data-src") | ||||||
|  |                     .let { link -> | ||||||
|  |                         invokeLocalSource(source, link, callback) | ||||||
|  |                     } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return true | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     data class Result( | ||||||
|  |         @JsonProperty("result") val result: ArrayList<Media>? = arrayListOf(), | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     data class Media( | ||||||
|  |         @JsonProperty("title") val title: String? = null, | ||||||
|  |         @JsonProperty("poster") val poster: String? = null, | ||||||
|  |         @JsonProperty("slug") val slug: String? = null, | ||||||
|  |         @JsonProperty("slug_prefix") val slugPrefix: String? = null, | ||||||
|  |     ) | ||||||
|  | } | ||||||
|  | @ -0,0 +1,14 @@ | ||||||
|  | 
 | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.lagradost.cloudstream3.plugins.CloudstreamPlugin | ||||||
|  | import com.lagradost.cloudstream3.plugins.Plugin | ||||||
|  | import android.content.Context | ||||||
|  | 
 | ||||||
|  | @CloudstreamPlugin | ||||||
|  | class HdfilmcehennemiPlugin: Plugin() { | ||||||
|  |     override fun load(context: Context) { | ||||||
|  |         // All providers should be added in this manner. Please don't edit the providers list directly. | ||||||
|  |         registerMainAPI(Hdfilmcehennemi()) | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										26
									
								
								IdlixProvider/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								IdlixProvider/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,26 @@ | ||||||
|  | // use an integer for version numbers | ||||||
|  | version = 2 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | cloudstream { | ||||||
|  |     language = "id" | ||||||
|  |     // All of these properties are optional, you can safely remove them | ||||||
|  | 
 | ||||||
|  |     // description = "Lorem Ipsum" | ||||||
|  |      authors = listOf("Hexated") | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * 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=94.103.82.88&sz=%size%" | ||||||
|  | } | ||||||
							
								
								
									
										2
									
								
								IdlixProvider/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								IdlixProvider/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,2 @@ | ||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <manifest package="com.hexated"/> | ||||||
							
								
								
									
										385
									
								
								IdlixProvider/src/main/kotlin/com/hexated/IdlixProvider.kt
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										385
									
								
								IdlixProvider/src/main/kotlin/com/hexated/IdlixProvider.kt
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,385 @@ | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.fasterxml.jackson.annotation.JsonProperty | ||||||
|  | import com.lagradost.cloudstream3.* | ||||||
|  | import com.lagradost.cloudstream3.LoadResponse.Companion.addActors | ||||||
|  | import com.lagradost.cloudstream3.LoadResponse.Companion.addTrailer | ||||||
|  | import com.lagradost.cloudstream3.mvvm.safeApiCall | ||||||
|  | import com.lagradost.cloudstream3.utils.* | ||||||
|  | import com.lagradost.cloudstream3.utils.AppUtils.tryParseJson | ||||||
|  | import org.jsoup.nodes.Element | ||||||
|  | import java.net.URI | ||||||
|  | 
 | ||||||
|  | class IdlixProvider : MainAPI() { | ||||||
|  |     override var mainUrl = "https://195.2.92.213" | ||||||
|  |     override var name = "Idlix" | ||||||
|  |     override val hasMainPage = true | ||||||
|  |     override var lang = "id" | ||||||
|  |     override val hasDownloadSupport = true | ||||||
|  |     override val supportedTypes = setOf( | ||||||
|  |         TvType.Movie, | ||||||
|  |         TvType.TvSeries, | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     override val mainPage = mainPageOf( | ||||||
|  |         "$mainUrl/trending/page/?get=movies" to "Trending Movies", | ||||||
|  |         "$mainUrl/trending/page/?get=tv" to "Trending TV Series", | ||||||
|  |         "$mainUrl/movie/page/" to "Movie Terbaru", | ||||||
|  |         "$mainUrl/tvseries/page/" to "TV Series Terbaru", | ||||||
|  |         "$mainUrl/season/page/" to "Season Terbaru", | ||||||
|  |         "$mainUrl/episode/page/" to "Episode Terbaru", | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     override suspend fun getMainPage( | ||||||
|  |         page: Int, | ||||||
|  |         request: MainPageRequest | ||||||
|  |     ): HomePageResponse { | ||||||
|  |         val url = request.data.split("?") | ||||||
|  |         val document = app.get("${url.first()}$page/?${url.lastOrNull()}").document | ||||||
|  |         val home = document.select("div.items.full article, div#archive-content article").mapNotNull { | ||||||
|  |             it.toSearchResult() | ||||||
|  |         } | ||||||
|  |         return newHomePageResponse(request.name, home) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun getProperLink(uri: String): String { | ||||||
|  |         return when { | ||||||
|  |             uri.contains("/episode/") -> { | ||||||
|  |                 var title = uri.substringAfter("$mainUrl/episode/") | ||||||
|  |                 title = Regex("(.+?)-season").find(title)?.groupValues?.get(1).toString() | ||||||
|  |                 "$mainUrl/tvseries/$title" | ||||||
|  |             } | ||||||
|  |             uri.contains("/season/") -> { | ||||||
|  |                 var title = uri.substringAfter("$mainUrl/season/") | ||||||
|  |                 title = Regex("(.+?)-season").find(title)?.groupValues?.get(1).toString() | ||||||
|  |                 "$mainUrl/tvseries/$title" | ||||||
|  |             } | ||||||
|  |             else -> { | ||||||
|  |                 uri | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun Element.toSearchResult(): SearchResponse { | ||||||
|  |         val title = this.selectFirst("h3 > a")!!.text().replace(Regex("\\(\\d{4}\\)"), "").trim() | ||||||
|  |         val href = getProperLink(this.selectFirst("h3 > a")!!.attr("href")) | ||||||
|  |         val posterUrl = this.select("div.poster > img").attr("src").toString() | ||||||
|  |         val quality = getQualityFromString(this.select("span.quality").text()) | ||||||
|  |         return newMovieSearchResponse(title, href, TvType.Movie) { | ||||||
|  |             this.posterUrl = posterUrl | ||||||
|  |             this.quality = quality | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun search(query: String): List<SearchResponse> { | ||||||
|  |         val link = "$mainUrl/search/$query" | ||||||
|  |         val document = app.get(link).document | ||||||
|  | 
 | ||||||
|  |         return document.select("div.result-item").map { | ||||||
|  |             val title = | ||||||
|  |                 it.selectFirst("div.title > a")!!.text().replace(Regex("\\(\\d{4}\\)"), "").trim() | ||||||
|  |             val href = getProperLink(it.selectFirst("div.title > a")!!.attr("href")) | ||||||
|  |             val posterUrl = it.selectFirst("img")!!.attr("src").toString() | ||||||
|  |             newMovieSearchResponse(title, href, TvType.TvSeries) { | ||||||
|  |                 this.posterUrl = posterUrl | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun load(url: String): LoadResponse { | ||||||
|  |         val document = app.get(url).document | ||||||
|  | 
 | ||||||
|  |         val title = | ||||||
|  |             document.selectFirst("div.data > h1")?.text()?.replace(Regex("\\(\\d{4}\\)"), "") | ||||||
|  |                 ?.trim().toString() | ||||||
|  |         val poster = document.select("div.poster > img").attr("src").toString() | ||||||
|  |         val tags = document.select("div.sgeneros > a").map { it.text() } | ||||||
|  | 
 | ||||||
|  |         val year = Regex(",\\s?(\\d+)").find( | ||||||
|  |             document.select("span.date").text().trim() | ||||||
|  |         )?.groupValues?.get(1).toString().toIntOrNull() | ||||||
|  |         val tvType = if (document.select("ul#section > li:nth-child(1)").text().contains("Episodes") | ||||||
|  |         ) TvType.TvSeries else TvType.Movie | ||||||
|  |         val description = document.select("div.wp-content > p").text().trim() | ||||||
|  |         val trailer = document.selectFirst("div.embed iframe")?.attr("src") | ||||||
|  |         val rating = | ||||||
|  |             document.selectFirst("span.dt_rating_vgs")?.text()?.toRatingInt() | ||||||
|  |         val actors = document.select("div.persons > div[itemprop=actor]").map { | ||||||
|  |             Actor(it.select("meta[itemprop=name]").attr("content"), it.select("img").attr("src")) | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         val recommendations = document.select("div.owl-item").map { | ||||||
|  |             val recName = | ||||||
|  |                 it.selectFirst("a")!!.attr("href").toString().removeSuffix("/").split("/").last() | ||||||
|  |             val recHref = it.selectFirst("a")!!.attr("href") | ||||||
|  |             val recPosterUrl = it.selectFirst("img")?.attr("src").toString() | ||||||
|  |             newTvSeriesSearchResponse(recName, recHref, TvType.TvSeries) { | ||||||
|  |                 this.posterUrl = recPosterUrl | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return if (tvType == TvType.TvSeries) { | ||||||
|  |             val episodes = document.select("ul.episodios > li").map { | ||||||
|  |                 val href = it.select("a").attr("href") | ||||||
|  |                 val name = fixTitle(it.select("div.episodiotitle > a").text().trim()) | ||||||
|  |                 val image = it.select("div.imagen > img").attr("src") | ||||||
|  |                 val episode = it.select("div.numerando").text().replace(" ", "").split("-").last() | ||||||
|  |                     .toIntOrNull() | ||||||
|  |                 val season = it.select("div.numerando").text().replace(" ", "").split("-").first() | ||||||
|  |                     .toIntOrNull() | ||||||
|  |                 Episode( | ||||||
|  |                     href, | ||||||
|  |                     name, | ||||||
|  |                     season, | ||||||
|  |                     episode, | ||||||
|  |                     image | ||||||
|  |                 ) | ||||||
|  |             } | ||||||
|  |             newTvSeriesLoadResponse(title, url, TvType.TvSeries, episodes) { | ||||||
|  |                 this.posterUrl = poster | ||||||
|  |                 this.year = year | ||||||
|  |                 this.plot = description | ||||||
|  |                 this.tags = tags | ||||||
|  |                 this.rating = rating | ||||||
|  |                 addActors(actors) | ||||||
|  |                 this.recommendations = recommendations | ||||||
|  |                 addTrailer(trailer) | ||||||
|  |             } | ||||||
|  |         } else { | ||||||
|  |             newMovieLoadResponse(title, url, TvType.Movie, url) { | ||||||
|  |                 this.posterUrl = poster | ||||||
|  |                 this.year = year | ||||||
|  |                 this.plot = description | ||||||
|  |                 this.tags = tags | ||||||
|  |                 this.rating = rating | ||||||
|  |                 addActors(actors) | ||||||
|  |                 this.recommendations = recommendations | ||||||
|  |                 addTrailer(trailer) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun getLanguage(str: String): String { | ||||||
|  |         return when { | ||||||
|  |             str.lowercase().contains("indonesia") || str.lowercase() | ||||||
|  |                 .contains("bahasa") -> "Indonesian" | ||||||
|  |             else -> str | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     data class ResponseHash( | ||||||
|  |         @JsonProperty("embed_url") val embed_url: String, | ||||||
|  |         @JsonProperty("type") val type: String?, | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     data class ResponseSource( | ||||||
|  |         @JsonProperty("hls") val hls: Boolean, | ||||||
|  |         @JsonProperty("videoSource") val videoSource: String, | ||||||
|  |         @JsonProperty("securedLink") val securedLink: String?, | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     data class Tracks( | ||||||
|  |         @JsonProperty("kind") val kind: String?, | ||||||
|  |         @JsonProperty("file") val file: String, | ||||||
|  |         @JsonProperty("label") val label: String?, | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     private suspend fun invokeLokalSource( | ||||||
|  |         url: String, | ||||||
|  |         subCallback: (SubtitleFile) -> Unit, | ||||||
|  |         sourceCallback: (ExtractorLink) -> Unit | ||||||
|  |     ) { | ||||||
|  |         val document = app.get(url, referer = "$mainUrl/").document | ||||||
|  |         val hash = url.split("/").last().substringAfter("data=") | ||||||
|  | 
 | ||||||
|  |         val m3uLink = app.post( | ||||||
|  |             url = "https://jeniusplay.com/player/index.php?data=$hash&do=getVideo", | ||||||
|  |             data = mapOf("hash" to hash, "r" to "$mainUrl/"), | ||||||
|  |             referer = url, | ||||||
|  |             headers = mapOf("X-Requested-With" to "XMLHttpRequest") | ||||||
|  |         ).parsed<ResponseSource>().videoSource | ||||||
|  | 
 | ||||||
|  |         M3u8Helper.generateM3u8( | ||||||
|  |             this.name, | ||||||
|  |             m3uLink, | ||||||
|  |             url, | ||||||
|  |         ).forEach(sourceCallback) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |         document.select("script").map { script -> | ||||||
|  |             if (script.data().contains("eval(function(p,a,c,k,e,d)")) { | ||||||
|  |                 val subData = | ||||||
|  |                     getAndUnpack(script.data()).substringAfter("\"tracks\":[").substringBefore("],") | ||||||
|  |                 tryParseJson<List<Tracks>>("[$subData]")?.map { subtitle -> | ||||||
|  |                     subCallback.invoke( | ||||||
|  |                         SubtitleFile( | ||||||
|  |                             getLanguage(subtitle.label!!), | ||||||
|  |                             subtitle.file | ||||||
|  |                         ) | ||||||
|  |                     ) | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     data class ResponseLaviolaSource( | ||||||
|  |         @JsonProperty("file") val file: String, | ||||||
|  |         @JsonProperty("label") val label: String?, | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     private suspend fun invokeLaviolaSource( | ||||||
|  |         url: String, | ||||||
|  |         subCallback: (SubtitleFile) -> Unit, | ||||||
|  |         sourceCallback: (ExtractorLink) -> Unit | ||||||
|  |     ) { | ||||||
|  |         val document = app.get(url, referer = "$mainUrl/").document | ||||||
|  |         val baseName = "Laviola" | ||||||
|  |         val baseUrl = "https://laviola.live/" | ||||||
|  |         document.select("script").map { script -> | ||||||
|  |             if (script.data().contains("var config = {")) { | ||||||
|  |                 val data = script.data().substringAfter("sources: [").substringBefore("],") | ||||||
|  |                 tryParseJson<List<ResponseLaviolaSource>>("[$data]")?.map { m3u -> | ||||||
|  |                     val m3uData = app.get(m3u.file, referer = baseUrl).text | ||||||
|  |                     val quality = | ||||||
|  |                         Regex("\\d{3,4}\\.m3u8").findAll(m3uData).map { it.value }.toList() | ||||||
|  |                     quality.forEach { | ||||||
|  |                         sourceCallback.invoke( | ||||||
|  |                             ExtractorLink( | ||||||
|  |                                 source = baseName, | ||||||
|  |                                 name = baseName, | ||||||
|  |                                 url = m3u.file.replace("video.m3u8", it), | ||||||
|  |                                 referer = baseUrl, | ||||||
|  |                                 quality = getQualityFromName("${it.replace(".m3u8", "")}p"), | ||||||
|  |                                 isM3u8 = true | ||||||
|  |                             ) | ||||||
|  |                         ) | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 val subData = script.data().substringAfter("tracks: [").substringBefore("],") | ||||||
|  |                 tryParseJson<List<Tracks>>("[$subData]")?.map { subtitle -> | ||||||
|  |                     subCallback.invoke( | ||||||
|  |                         SubtitleFile( | ||||||
|  |                             getLanguage(subtitle.label!!), | ||||||
|  |                             (if (subtitle.kind!!.contains("captions")) subtitle.file else null)!! | ||||||
|  |                         ) | ||||||
|  |                     ) | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private data class Captions( | ||||||
|  |         @JsonProperty("id") val id: String, | ||||||
|  |         @JsonProperty("hash") val hash: String, | ||||||
|  |         @JsonProperty("language") val language: String, | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     private data class Data( | ||||||
|  |         @JsonProperty("file") val file: String, | ||||||
|  |         @JsonProperty("label") val label: String, | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     private data class Player( | ||||||
|  |         @JsonProperty("poster_file") val poster_file: String, | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     private data class ResponseCdn( | ||||||
|  |         @JsonProperty("success") val success: Boolean, | ||||||
|  |         @JsonProperty("player") val player: Player, | ||||||
|  |         @JsonProperty("data") val data: List<Data>?, | ||||||
|  |         @JsonProperty("captions") val captions: List<Captions>? | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     private suspend fun invokeCdnSource( | ||||||
|  |         url: String, | ||||||
|  |         subCallback: (SubtitleFile) -> Unit, | ||||||
|  |         sourceCallback: (ExtractorLink) -> Unit | ||||||
|  |     ) { | ||||||
|  |         val domainUrl = "https://cdnplayer.online" | ||||||
|  |         val id = url.trimEnd('/').split("/").last() | ||||||
|  |         val sources = app.post( | ||||||
|  |             url = "$domainUrl/api/source/$id", | ||||||
|  |             data = mapOf("r" to mainUrl, "d" to URI(url).host) | ||||||
|  |         ).parsed<ResponseCdn>() | ||||||
|  | 
 | ||||||
|  |         sources.data?.map { | ||||||
|  |             sourceCallback.invoke( | ||||||
|  |                 ExtractorLink( | ||||||
|  |                     name, | ||||||
|  |                     "Cdnplayer", | ||||||
|  |                     fixUrl(it.file), | ||||||
|  |                     referer = url, | ||||||
|  |                     quality = getQualityFromName(it.label) | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |         } | ||||||
|  |         val userData = sources.player.poster_file.split("/")[2] | ||||||
|  |         sources.captions?.map { subtitle -> | ||||||
|  |             subCallback.invoke( | ||||||
|  |                 SubtitleFile( | ||||||
|  |                     getLanguage(subtitle.language), | ||||||
|  |                     "$domainUrl/asset/userdata/$userData/caption/${subtitle.hash}/${subtitle.id}.srt" | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun loadLinks( | ||||||
|  |         data: String, | ||||||
|  |         isCasting: Boolean, | ||||||
|  |         subtitleCallback: (SubtitleFile) -> Unit, | ||||||
|  |         callback: (ExtractorLink) -> Unit | ||||||
|  |     ): Boolean { | ||||||
|  | 
 | ||||||
|  |         val document = app.get(data).document | ||||||
|  |         val id = document.select("meta#dooplay-ajax-counter").attr("data-postid") | ||||||
|  |         val type = if (data.contains("/movie/")) "movie" else "tv" | ||||||
|  | 
 | ||||||
|  |         document.select("ul#playeroptionsul > li").map { | ||||||
|  |             it.attr("data-nume") | ||||||
|  |         }.apmap { nume -> | ||||||
|  |             safeApiCall { | ||||||
|  |                 var source = app.post( | ||||||
|  |                     url = "$mainUrl/wp-admin/admin-ajax.php", | ||||||
|  |                     data = mapOf( | ||||||
|  |                         "action" to "doo_player_ajax", | ||||||
|  |                         "post" to id, | ||||||
|  |                         "nume" to nume, | ||||||
|  |                         "type" to type | ||||||
|  |                     ) | ||||||
|  |                 ).parsed<ResponseHash>().embed_url | ||||||
|  | 
 | ||||||
|  |                 when { | ||||||
|  |                     source.startsWith("https://jeniusplay.com") -> invokeLokalSource( | ||||||
|  |                         source, | ||||||
|  |                         subtitleCallback, | ||||||
|  |                         callback | ||||||
|  |                     ) | ||||||
|  |                     source.startsWith("https://laviola.live") -> invokeLaviolaSource( | ||||||
|  |                         source, | ||||||
|  |                         subtitleCallback, | ||||||
|  |                         callback | ||||||
|  |                     ) | ||||||
|  |                     source.startsWith("https://cdnplayer.online") -> invokeCdnSource( | ||||||
|  |                         source, | ||||||
|  |                         subtitleCallback, | ||||||
|  |                         callback | ||||||
|  |                     ) | ||||||
|  |                     else -> { | ||||||
|  |                         if (source.startsWith("https://uservideo.xyz")) { | ||||||
|  |                             source = app.get(source).document.select("iframe").attr("src") | ||||||
|  |                         } | ||||||
|  |                         loadExtractor(source, data, subtitleCallback, callback) | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return true | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | @ -0,0 +1,14 @@ | ||||||
|  | 
 | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.lagradost.cloudstream3.plugins.CloudstreamPlugin | ||||||
|  | import com.lagradost.cloudstream3.plugins.Plugin | ||||||
|  | import android.content.Context | ||||||
|  | 
 | ||||||
|  | @CloudstreamPlugin | ||||||
|  | class IdlixProviderPlugin: Plugin() { | ||||||
|  |     override fun load(context: Context) { | ||||||
|  |         // All providers should be added in this manner. Please don't edit the providers list directly. | ||||||
|  |         registerMainAPI(IdlixProvider()) | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										28
									
								
								KisskhProvider/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								KisskhProvider/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,28 @@ | ||||||
|  | // use an integer for version numbers | ||||||
|  | version = 3 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | cloudstream { | ||||||
|  |     language = "en" | ||||||
|  |     // All of these properties are optional, you can safely remove them | ||||||
|  | 
 | ||||||
|  |     // description = "Lorem Ipsum" | ||||||
|  |      authors = listOf("Hexated") | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Status int as the following: | ||||||
|  |      * 0: Down | ||||||
|  |      * 1: Ok | ||||||
|  |      * 2: Slow | ||||||
|  |      * 3: Beta only | ||||||
|  |      * */ | ||||||
|  |     status = 1 // will be 3 if unspecified | ||||||
|  |     tvTypes = listOf( | ||||||
|  |         "AsianDrama", | ||||||
|  |         "TvSeries", | ||||||
|  |         "Anime", | ||||||
|  |         "Movie", | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     iconUrl = "https://www.google.com/s2/favicons?domain=kisskh.me&sz=%size%" | ||||||
|  | } | ||||||
							
								
								
									
										2
									
								
								KisskhProvider/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								KisskhProvider/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,2 @@ | ||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <manifest package="com.hexated"/> | ||||||
							
								
								
									
										217
									
								
								KisskhProvider/src/main/kotlin/com/hexated/KisskhProvider.kt
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										217
									
								
								KisskhProvider/src/main/kotlin/com/hexated/KisskhProvider.kt
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,217 @@ | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.fasterxml.jackson.annotation.JsonProperty | ||||||
|  | import com.lagradost.cloudstream3.* | ||||||
|  | import com.lagradost.cloudstream3.mvvm.safeApiCall | ||||||
|  | import com.lagradost.cloudstream3.utils.AppUtils.parseJson | ||||||
|  | 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.M3u8Helper | ||||||
|  | import com.lagradost.cloudstream3.utils.loadExtractor | ||||||
|  | import java.util.ArrayList | ||||||
|  | 
 | ||||||
|  | class KisskhProvider : MainAPI() { | ||||||
|  |     override var mainUrl = "https://kisskh.me" | ||||||
|  |     override var name = "Kisskh" | ||||||
|  |     override val hasMainPage = true | ||||||
|  |     override val hasDownloadSupport = true | ||||||
|  |     override val supportedTypes = setOf( | ||||||
|  |         TvType.AsianDrama, | ||||||
|  |         TvType.Anime | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     override val mainPage = mainPageOf( | ||||||
|  |         "&type=2&sub=0&country=2&status=0&order=1" to "Movie Popular", | ||||||
|  |         "&type=2&sub=0&country=2&status=0&order=2" to "Movie Last Update", | ||||||
|  |         "&type=1&sub=0&country=2&status=0&order=1" to "TVSeries Popular", | ||||||
|  |         "&type=1&sub=0&country=2&status=0&order=2" to "TVSeries Last Update", | ||||||
|  |         "&type=3&sub=0&country=0&status=0&order=1" to "Anime Popular", | ||||||
|  |         "&type=3&sub=0&country=0&status=0&order=2" to "Anime Last Update", | ||||||
|  |         "&type=4&sub=0&country=0&status=0&order=1" to "Hollywood Popular", | ||||||
|  |         "&type=4&sub=0&country=0&status=0&order=2" to "Hollywood Last Update", | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     override suspend fun getMainPage( | ||||||
|  |         page: Int, | ||||||
|  |         request: MainPageRequest | ||||||
|  |     ): HomePageResponse { | ||||||
|  |         val home = app.get("$mainUrl/api/DramaList/List?page=$page${request.data}") | ||||||
|  |             .parsedSafe<Responses>()?.data | ||||||
|  |             ?.mapNotNull { media -> | ||||||
|  |                 media.toSearchResponse() | ||||||
|  |             } ?: throw ErrorLoadingException("Invalid Json reponse") | ||||||
|  |         return newHomePageResponse( | ||||||
|  |             list = HomePageList( | ||||||
|  |                 name = request.name, | ||||||
|  |                 list = home, | ||||||
|  |                 isHorizontalImages = true | ||||||
|  |             ), | ||||||
|  |             hasNext = true | ||||||
|  |         ) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun Media.toSearchResponse(): SearchResponse? { | ||||||
|  | 
 | ||||||
|  |         return newAnimeSearchResponse( | ||||||
|  |             title ?: return null, | ||||||
|  |             "$title/$id", | ||||||
|  |             TvType.TvSeries, | ||||||
|  |         ) { | ||||||
|  |             this.posterUrl = thumbnail | ||||||
|  |             addSub(episodesCount) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun search(query: String): List<SearchResponse> { | ||||||
|  |         val searchResponse = | ||||||
|  |             app.get("$mainUrl/api/DramaList/Search?q=$query&type=0", referer = "$mainUrl/").text | ||||||
|  |         return tryParseJson<ArrayList<Media>>(searchResponse)?.mapNotNull { media -> | ||||||
|  |             media.toSearchResponse() | ||||||
|  |         } ?: throw ErrorLoadingException("Invalid Json reponse") | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun getTitle(str: String): String { | ||||||
|  |         return str.replace(Regex("[^a-zA-Z0-9]"), "-") | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun load(url: String): LoadResponse? { | ||||||
|  |         val id = url.split("/") | ||||||
|  |         val res = app.get( | ||||||
|  |             "$mainUrl/api/DramaList/Drama/${id.last()}?isq=false", | ||||||
|  |             referer = "$mainUrl/Drama/${ | ||||||
|  |                 getTitle(id.first()) | ||||||
|  |             }?id=${id.last()}" | ||||||
|  |         ).parsedSafe<MediaDetail>() | ||||||
|  |             ?: throw ErrorLoadingException("Invalid Json reponse") | ||||||
|  | 
 | ||||||
|  |         val episodes = res.episodes?.map { eps -> | ||||||
|  |             Episode( | ||||||
|  |                 data = Data(res.title, eps.number, res.id, eps.id).toJson(), | ||||||
|  |                 episode = eps.number | ||||||
|  |             ) | ||||||
|  |         } ?: throw ErrorLoadingException("No Episode") | ||||||
|  | 
 | ||||||
|  |         return newTvSeriesLoadResponse( | ||||||
|  |             res.title ?: return null, | ||||||
|  |             url, | ||||||
|  |             if (res.type == "Movie" || episodes.size == 1) TvType.Movie else TvType.TvSeries, | ||||||
|  |             episodes | ||||||
|  |         ) { | ||||||
|  |             this.posterUrl = res.thumbnail | ||||||
|  |             this.year = res.releaseDate?.split("-")?.first()?.toIntOrNull() | ||||||
|  |             this.plot = res.description | ||||||
|  |             this.tags = listOf("${res.country}", "${res.status}", "${res.type}") | ||||||
|  |             this.showStatus = when (res.status) { | ||||||
|  |                 "Completed" -> ShowStatus.Completed | ||||||
|  |                 "Ongoing" -> ShowStatus.Ongoing | ||||||
|  |                 else -> null | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun getLanguage(str: String): String { | ||||||
|  |         return when (str) { | ||||||
|  |             "Indonesia" -> "Indonesian" | ||||||
|  |             else -> str | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun loadLinks( | ||||||
|  |         data: String, | ||||||
|  |         isCasting: Boolean, | ||||||
|  |         subtitleCallback: (SubtitleFile) -> Unit, | ||||||
|  |         callback: (ExtractorLink) -> Unit | ||||||
|  |     ): Boolean { | ||||||
|  | 
 | ||||||
|  |         val loadData = parseJson<Data>(data) | ||||||
|  | 
 | ||||||
|  |         app.get( | ||||||
|  |             "$mainUrl/api/DramaList/Episode/${loadData.epsId}.png?err=false&ts=&time=", | ||||||
|  |             referer = "$mainUrl/Drama/${getTitle("${loadData.title}")}/Episode-${loadData.eps}?id=${loadData.id}&ep=${loadData.epsId}&page=0&pageSize=100" | ||||||
|  |         ).parsedSafe<Sources>()?.let { source -> | ||||||
|  |             listOf(source.video, source.thirdParty).apmap { link -> | ||||||
|  |                 safeApiCall { | ||||||
|  |                     if (link?.contains(".m3u8") == true) { | ||||||
|  |                         M3u8Helper.generateM3u8( | ||||||
|  |                             this.name, | ||||||
|  |                             link, | ||||||
|  |                             referer = "$mainUrl/", | ||||||
|  |                             headers = mapOf("Origin" to mainUrl) | ||||||
|  |                         ).forEach(callback) | ||||||
|  |                     } else { | ||||||
|  |                         loadExtractor( | ||||||
|  |                             link?.substringBefore("=http") ?: return@safeApiCall, | ||||||
|  |                             "$mainUrl/", | ||||||
|  |                             subtitleCallback, | ||||||
|  |                             callback | ||||||
|  |                         ) | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         // parsedSafe doesn't work in <List<Object>> | ||||||
|  |         app.get("$mainUrl/api/Sub/${loadData.epsId}").text.let { res -> | ||||||
|  |             tryParseJson<List<Subtitle>>(res)?.map { sub -> | ||||||
|  |                 subtitleCallback.invoke( | ||||||
|  |                     SubtitleFile( | ||||||
|  |                         getLanguage(sub.label ?: return@map), | ||||||
|  |                         sub.src ?: return@map | ||||||
|  |                     ) | ||||||
|  |                 ) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return true | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     data class Data( | ||||||
|  |         val title: String?, | ||||||
|  |         val eps: Int?, | ||||||
|  |         val id: Int?, | ||||||
|  |         val epsId: Int?, | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     data class Sources( | ||||||
|  |         @JsonProperty("Video") val video: String?, | ||||||
|  |         @JsonProperty("ThirdParty") val thirdParty: String?, | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     data class Subtitle( | ||||||
|  |         @JsonProperty("src") val src: String?, | ||||||
|  |         @JsonProperty("label") val label: String?, | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     data class Responses( | ||||||
|  |         @JsonProperty("data") val data: ArrayList<Media>? = arrayListOf(), | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     data class Media( | ||||||
|  |         @JsonProperty("episodesCount") val episodesCount: Int?, | ||||||
|  |         @JsonProperty("thumbnail") val thumbnail: String?, | ||||||
|  |         @JsonProperty("id") val id: Int?, | ||||||
|  |         @JsonProperty("title") val title: String?, | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     data class Episodes( | ||||||
|  |         @JsonProperty("id") val id: Int?, | ||||||
|  |         @JsonProperty("number") val number: Int?, | ||||||
|  |         @JsonProperty("sub") val sub: Int?, | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     data class MediaDetail( | ||||||
|  |         @JsonProperty("description") val description: String?, | ||||||
|  |         @JsonProperty("releaseDate") val releaseDate: String?, | ||||||
|  |         @JsonProperty("status") val status: String?, | ||||||
|  |         @JsonProperty("type") val type: String?, | ||||||
|  |         @JsonProperty("country") val country: String?, | ||||||
|  |         @JsonProperty("episodes") val episodes: ArrayList<Episodes>? = arrayListOf(), | ||||||
|  |         @JsonProperty("thumbnail") val thumbnail: String?, | ||||||
|  |         @JsonProperty("id") val id: Int?, | ||||||
|  |         @JsonProperty("title") val title: String?, | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | @ -0,0 +1,14 @@ | ||||||
|  | 
 | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.lagradost.cloudstream3.plugins.CloudstreamPlugin | ||||||
|  | import com.lagradost.cloudstream3.plugins.Plugin | ||||||
|  | import android.content.Context | ||||||
|  | 
 | ||||||
|  | @CloudstreamPlugin | ||||||
|  | class KisskhProviderPlugin: Plugin() { | ||||||
|  |     override fun load(context: Context) { | ||||||
|  |         // All providers should be added in this manner. Please don't edit the providers list directly. | ||||||
|  |         registerMainAPI(KisskhProvider()) | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										27
									
								
								KuramanimeProvider/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								KuramanimeProvider/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,27 @@ | ||||||
|  | // use an integer for version numbers | ||||||
|  | version = 2 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | cloudstream { | ||||||
|  |     language = "id" | ||||||
|  |     // All of these properties are optional, you can safely remove them | ||||||
|  | 
 | ||||||
|  |     // description = "Lorem Ipsum" | ||||||
|  |      authors = listOf("Hexated") | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * 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=kuramanime.com&sz=%size%" | ||||||
|  | } | ||||||
							
								
								
									
										2
									
								
								KuramanimeProvider/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								KuramanimeProvider/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,2 @@ | ||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <manifest package="com.hexated"/> | ||||||
|  | @ -0,0 +1,164 @@ | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.lagradost.cloudstream3.* | ||||||
|  | import com.lagradost.cloudstream3.mvvm.suspendSafeApiCall | ||||||
|  | import com.lagradost.cloudstream3.utils.ExtractorLink | ||||||
|  | import org.jsoup.Jsoup | ||||||
|  | import org.jsoup.nodes.Element | ||||||
|  | 
 | ||||||
|  | class KuramanimeProvider : MainAPI() { | ||||||
|  |     override var mainUrl = "https://kuramanime.com" | ||||||
|  |     override var name = "Kuramanime" | ||||||
|  |     override val hasQuickSearch = false | ||||||
|  |     override val hasMainPage = true | ||||||
|  |     override var lang = "id" | ||||||
|  |     override val hasDownloadSupport = true | ||||||
|  | 
 | ||||||
|  |     override val supportedTypes = setOf( | ||||||
|  |         TvType.Anime, | ||||||
|  |         TvType.AnimeMovie, | ||||||
|  |         TvType.OVA | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     companion object { | ||||||
|  |         fun getStatus(t: String): ShowStatus { | ||||||
|  |             return when (t) { | ||||||
|  |                 "Selesai Tayang" -> ShowStatus.Completed | ||||||
|  |                 "Sedang Tayang" -> ShowStatus.Ongoing | ||||||
|  |                 else -> ShowStatus.Completed | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override val mainPage = mainPageOf( | ||||||
|  |         "$mainUrl/anime/ongoing?order_by=updated&page=" to "Sedang Tayang", | ||||||
|  |         "$mainUrl/anime/finished?order_by=updated&page=" to "Selesai Tayang", | ||||||
|  |         "$mainUrl/properties/season/summer-2022?order_by=most_viewed&page=" to "Dilihat Terbanyak Musim Ini", | ||||||
|  |         "$mainUrl/anime/movie?order_by=updated&page=" to "Film Layar Lebar", | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     override suspend fun getMainPage( | ||||||
|  |         page: Int, | ||||||
|  |         request: MainPageRequest | ||||||
|  |     ): HomePageResponse { | ||||||
|  |         val document = app.get(request.data + page).document | ||||||
|  | 
 | ||||||
|  |         val home = document.select("div.col-lg-4.col-md-6.col-sm-6").mapNotNull { | ||||||
|  |             it.toSearchResult() | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return newHomePageResponse(request.name, home) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun getProperAnimeLink(uri: String): String { | ||||||
|  |         return if (uri.contains("/episode")) { | ||||||
|  |             Regex("(.*)/episode/.+").find(uri)?.groupValues?.get(1).toString() + "/" | ||||||
|  |         } else { | ||||||
|  |             uri | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun Element.toSearchResult(): AnimeSearchResponse? { | ||||||
|  |         val href = getProperAnimeLink(fixUrl(this.selectFirst("a")!!.attr("href"))) | ||||||
|  |         val title = this.selectFirst("h5 a")?.text() ?: return null | ||||||
|  |         val posterUrl = fixUrl(this.select("div.product__item__pic.set-bg").attr("data-setbg")) | ||||||
|  |         val episode = Regex("([0-9*])\\s?/").find( | ||||||
|  |             this.select("div.ep span").text() | ||||||
|  |         )?.groupValues?.getOrNull(1)?.toIntOrNull() | ||||||
|  | 
 | ||||||
|  |         return newAnimeSearchResponse(title, href, TvType.Anime) { | ||||||
|  |             this.posterUrl = posterUrl | ||||||
|  |             addSub(episode) | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun search(query: String): List<SearchResponse> { | ||||||
|  |         val link = "$mainUrl/anime?search=$query&order_by=latest" | ||||||
|  |         val document = app.get(link).document | ||||||
|  | 
 | ||||||
|  |         return document.select("div#animeList div.col-lg-4.col-md-6.col-sm-6").mapNotNull { | ||||||
|  |             it.toSearchResult() | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun load(url: String): LoadResponse { | ||||||
|  |         val document = app.get(url).document | ||||||
|  | 
 | ||||||
|  |         val title = document.selectFirst(".anime__details__title > h3")!!.text().trim() | ||||||
|  |         val poster = document.selectFirst(".anime__details__pic")?.attr("data-setbg") | ||||||
|  |         val tags = | ||||||
|  |             document.select("div.anime__details__widget > div > div:nth-child(2) > ul > li:nth-child(1)") | ||||||
|  |                 .text().trim().replace("Genre: ", "").split(", ") | ||||||
|  | 
 | ||||||
|  |         val year = Regex("[^0-9]").replace( | ||||||
|  |             document.select("div.anime__details__widget > div > div:nth-child(1) > ul > li:nth-child(5)") | ||||||
|  |                 .text().trim().replace("Musim: ", ""), "" | ||||||
|  |         ).toIntOrNull() | ||||||
|  |         val status = getStatus( | ||||||
|  |             document.select("div.anime__details__widget > div > div:nth-child(1) > ul > li:nth-child(3)") | ||||||
|  |                 .text().trim().replace("Status: ", "") | ||||||
|  |         ) | ||||||
|  |         val description = document.select(".anime__details__text > p").text().trim() | ||||||
|  | 
 | ||||||
|  |         val episodes = | ||||||
|  |             Jsoup.parse(document.select("#episodeLists").attr("data-content")).select("a").map { | ||||||
|  |                 val name = it.text().trim() | ||||||
|  |                 val link = it.attr("href") | ||||||
|  |                 Episode(link, name) | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |         val recommendations = document.select("div#randomList > a").mapNotNull { | ||||||
|  |             val epHref = it.attr("href") | ||||||
|  |             val epTitle = it.select("h5.sidebar-title-h5.px-2.py-2").text() | ||||||
|  |             val epPoster = it.select(".product__sidebar__view__item.set-bg").attr("data-setbg") | ||||||
|  | 
 | ||||||
|  |             newAnimeSearchResponse(epTitle, epHref, TvType.Anime) { | ||||||
|  |                 this.posterUrl = epPoster | ||||||
|  |                 addDubStatus(dubExist = false, subExist = true) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return newAnimeLoadResponse(title, url, TvType.Anime) { | ||||||
|  |             engName = title | ||||||
|  |             posterUrl = poster | ||||||
|  |             this.year = year | ||||||
|  |             addEpisodes(DubStatus.Subbed, episodes) | ||||||
|  |             showStatus = status | ||||||
|  |             plot = description | ||||||
|  |             this.tags = tags | ||||||
|  |             this.recommendations = recommendations | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun loadLinks( | ||||||
|  |         data: String, | ||||||
|  |         isCasting: Boolean, | ||||||
|  |         subtitleCallback: (SubtitleFile) -> Unit, | ||||||
|  |         callback: (ExtractorLink) -> Unit | ||||||
|  |     ): Boolean { | ||||||
|  |         val servers = app.get(data).document | ||||||
|  |         servers.select("video#player > source").map { | ||||||
|  |             suspendSafeApiCall { | ||||||
|  |                 val url = it.attr("src") | ||||||
|  |                 val quality = it.attr("size").toInt() | ||||||
|  |                 callback.invoke( | ||||||
|  |                     ExtractorLink( | ||||||
|  |                         name, | ||||||
|  |                         name, | ||||||
|  |                         url, | ||||||
|  |                         referer = "$mainUrl/", | ||||||
|  |                         quality = quality, | ||||||
|  |                         headers = mapOf( | ||||||
|  |                             "Range" to "bytes=0-" | ||||||
|  |                         ) | ||||||
|  |                     ) | ||||||
|  |                 ) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return true | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | @ -0,0 +1,14 @@ | ||||||
|  | 
 | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.lagradost.cloudstream3.plugins.CloudstreamPlugin | ||||||
|  | import com.lagradost.cloudstream3.plugins.Plugin | ||||||
|  | import android.content.Context | ||||||
|  | 
 | ||||||
|  | @CloudstreamPlugin | ||||||
|  | class KuramanimeProviderPlugin: Plugin() { | ||||||
|  |     override fun load(context: Context) { | ||||||
|  |         // All providers should be added in this manner. Please don't edit the providers list directly. | ||||||
|  |         registerMainAPI(KuramanimeProvider()) | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										27
									
								
								KuronimeProvider/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								KuronimeProvider/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,27 @@ | ||||||
|  | // use an integer for version numbers | ||||||
|  | version = 2 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | cloudstream { | ||||||
|  |     language = "id" | ||||||
|  |     // All of these properties are optional, you can safely remove them | ||||||
|  | 
 | ||||||
|  |     // description = "Lorem Ipsum" | ||||||
|  |      authors = listOf("Hexated") | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * 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=45.12.2.2&sz=%size%" | ||||||
|  | } | ||||||
							
								
								
									
										2
									
								
								KuronimeProvider/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								KuronimeProvider/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,2 @@ | ||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <manifest package="com.hexated"/> | ||||||
							
								
								
									
										197
									
								
								KuronimeProvider/src/main/kotlin/com/hexated/KuronimeProvider.kt
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										197
									
								
								KuronimeProvider/src/main/kotlin/com/hexated/KuronimeProvider.kt
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,197 @@ | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.lagradost.cloudstream3.* | ||||||
|  | import com.lagradost.cloudstream3.LoadResponse.Companion.addTrailer | ||||||
|  | import com.lagradost.cloudstream3.mvvm.safeApiCall | ||||||
|  | import com.lagradost.cloudstream3.utils.ExtractorLink | ||||||
|  | import com.lagradost.cloudstream3.utils.getQualityFromName | ||||||
|  | import com.lagradost.cloudstream3.utils.loadExtractor | ||||||
|  | import org.jsoup.Jsoup | ||||||
|  | import org.jsoup.nodes.Element | ||||||
|  | 
 | ||||||
|  | class KuronimeProvider : MainAPI() { | ||||||
|  |     override var mainUrl = "https://45.12.2.2" | ||||||
|  |     override var name = "Kuronime" | ||||||
|  |     override val hasQuickSearch = false | ||||||
|  |     override val hasMainPage = true | ||||||
|  |     override var lang = "id" | ||||||
|  |     override val hasDownloadSupport = true | ||||||
|  | 
 | ||||||
|  |     override val supportedTypes = setOf( | ||||||
|  |         TvType.Anime, | ||||||
|  |         TvType.AnimeMovie, | ||||||
|  |         TvType.OVA | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     companion object { | ||||||
|  |         fun getType(t: String): TvType { | ||||||
|  |             return if (t.contains("OVA") || t.contains("Special")) TvType.OVA | ||||||
|  |             else if (t.contains("Movie")) TvType.AnimeMovie | ||||||
|  |             else TvType.Anime | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         fun getStatus(t: String): ShowStatus { | ||||||
|  |             return when (t) { | ||||||
|  |                 "Completed" -> ShowStatus.Completed | ||||||
|  |                 "Ongoing" -> ShowStatus.Ongoing | ||||||
|  |                 else -> ShowStatus.Completed | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override val mainPage = mainPageOf( | ||||||
|  |         "$mainUrl/page/" to "New Episodes", | ||||||
|  |         "$mainUrl/popular-anime/page/" to "Popular Anime", | ||||||
|  |         "$mainUrl/movies/page/" to "Movies", | ||||||
|  |         "$mainUrl/genres/donghua/page/" to "Donghua", | ||||||
|  |         "$mainUrl/live-action/page/" to "Live Action", | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     override suspend fun getMainPage( | ||||||
|  |         page: Int, | ||||||
|  |         request: MainPageRequest | ||||||
|  |     ): HomePageResponse { | ||||||
|  |         val document = app.get(request.data + page).document | ||||||
|  |         val home = document.select("article").map { | ||||||
|  |             it.toSearchResult() | ||||||
|  |         } | ||||||
|  |         return newHomePageResponse(request.name, home) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun getProperAnimeLink(uri: String): String { | ||||||
|  |         return if (uri.contains("/anime/")) { | ||||||
|  |             uri | ||||||
|  |         } else { | ||||||
|  |             var title = uri.substringAfter("$mainUrl/") | ||||||
|  |             title = when { | ||||||
|  |                 (title.contains("-episode")) && !(title.contains("-movie")) -> Regex("nonton-(.+)-episode").find( | ||||||
|  |                     title | ||||||
|  |                 )?.groupValues?.get(1).toString() | ||||||
|  |                 (title.contains("-movie")) -> Regex("nonton-(.+)-movie").find(title)?.groupValues?.get( | ||||||
|  |                     1 | ||||||
|  |                 ).toString() | ||||||
|  |                 else -> title | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             "$mainUrl/anime/$title" | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun Element.toSearchResult(): AnimeSearchResponse { | ||||||
|  |         val href = getProperAnimeLink(fixUrlNull(this.selectFirst("a")?.attr("href")).toString()) | ||||||
|  |         val title = this.select(".bsuxtt, .tt > h4").text().trim() | ||||||
|  |         val posterUrl = fixUrlNull( | ||||||
|  |             this.selectFirst("div.view,div.bt")?.nextElementSibling()?.select("img") | ||||||
|  |                 ?.attr("data-src") | ||||||
|  |         ) | ||||||
|  |         val epNum = this.select(".ep").text().replace(Regex("[^0-9]"), "").trim().toIntOrNull() | ||||||
|  |         val tvType = getType(this.selectFirst(".bt > span")?.text().toString()) | ||||||
|  |         return newAnimeSearchResponse(title, href, tvType) { | ||||||
|  |             this.posterUrl = posterUrl | ||||||
|  |             addSub(epNum) | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun search(query: String): List<SearchResponse> { | ||||||
|  |         val link = "$mainUrl/?s=$query" | ||||||
|  |         val document = app.get(link).document | ||||||
|  | 
 | ||||||
|  |         return document.select("article.bs").map { | ||||||
|  |             it.toSearchResult() | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun load(url: String): LoadResponse { | ||||||
|  |         val document = app.get(url).document | ||||||
|  | 
 | ||||||
|  |         val title = document.selectFirst(".entry-title")?.text().toString().trim() | ||||||
|  |         val poster = document.selectFirst("div.l[itemprop=image] > img")?.attr("data-src") | ||||||
|  |         val tags = document.select(".infodetail > ul > li:nth-child(2) > a").map { it.text() } | ||||||
|  |         val type = getType( | ||||||
|  |             document.selectFirst(".infodetail > ul > li:nth-child(7)")?.ownText()?.trim().toString() | ||||||
|  |         ) | ||||||
|  |         val trailer = document.selectFirst("div.tply iframe")?.attr("data-src") | ||||||
|  |         val year = Regex("\\d, ([0-9]*)").find( | ||||||
|  |             document.select(".infodetail > ul > li:nth-child(5)").text() | ||||||
|  |         )?.groupValues?.get(1)?.toIntOrNull() | ||||||
|  |         val status = getStatus( | ||||||
|  |             document.selectFirst(".infodetail > ul > li:nth-child(3)")!!.ownText() | ||||||
|  |                 .replace(Regex("\\W"), "") | ||||||
|  |         ) | ||||||
|  |         val description = document.select("span.const > p").text() | ||||||
|  | 
 | ||||||
|  |         val episodes = document.select("div.bixbox.bxcl > ul > li").map { | ||||||
|  |             val name = it.selectFirst("a")?.text()?.trim() | ||||||
|  |             val episode = | ||||||
|  |                 it.selectFirst("a")?.text()?.trim()?.replace("Episode", "")?.trim()?.toIntOrNull() | ||||||
|  |             val link = it.selectFirst("a")!!.attr("href") | ||||||
|  |             Episode(link, name = name, episode = episode) | ||||||
|  |         }.reversed() | ||||||
|  | 
 | ||||||
|  |         return newAnimeLoadResponse(title, url, type) { | ||||||
|  |             engName = title | ||||||
|  |             posterUrl = poster | ||||||
|  |             this.year = year | ||||||
|  |             addEpisodes(DubStatus.Subbed, episodes) | ||||||
|  |             showStatus = status | ||||||
|  |             plot = description | ||||||
|  |             addTrailer(trailer) | ||||||
|  |             this.tags = tags | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private suspend fun invokeKuroSource( | ||||||
|  |         url: String, | ||||||
|  |         sourceCallback: (ExtractorLink) -> Unit | ||||||
|  |     ) { | ||||||
|  |         val doc = app.get(url, referer = "${mainUrl}/").document | ||||||
|  | 
 | ||||||
|  |         doc.select("script").map { script -> | ||||||
|  |             if (script.data().contains("function jalankan_jwp() {")) { | ||||||
|  |                 val data = script.data() | ||||||
|  |                 val doma = data.substringAfter("var doma = \"").substringBefore("\";") | ||||||
|  |                 val token = data.substringAfter("var token = \"").substringBefore("\";") | ||||||
|  |                 val pat = data.substringAfter("var pat = \"").substringBefore("\";") | ||||||
|  |                 val link = "$doma$token$pat/index.m3u8" | ||||||
|  |                 val quality = | ||||||
|  |                     Regex("\\d{3,4}p").find(doc.select("title").text())?.groupValues?.get(0) | ||||||
|  | 
 | ||||||
|  |                 sourceCallback.invoke( | ||||||
|  |                     ExtractorLink( | ||||||
|  |                         this.name, | ||||||
|  |                         this.name, | ||||||
|  |                         link, | ||||||
|  |                         referer = "https://animeku.org/", | ||||||
|  |                         quality = getQualityFromName(quality), | ||||||
|  |                         headers = mapOf("Origin" to "https://animeku.org"), | ||||||
|  |                         isM3u8 = true | ||||||
|  |                     ) | ||||||
|  |                 ) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun loadLinks( | ||||||
|  |         data: String, | ||||||
|  |         isCasting: Boolean, | ||||||
|  |         subtitleCallback: (SubtitleFile) -> Unit, | ||||||
|  |         callback: (ExtractorLink) -> Unit | ||||||
|  |     ): Boolean { | ||||||
|  |         val document = app.get(data).document | ||||||
|  |         val sources = document.select(".mobius > .mirror > option").mapNotNull { | ||||||
|  |             fixUrl(Jsoup.parse(base64Decode(it.attr("value"))).select("iframe").attr("data-src")) | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         sources.apmap { | ||||||
|  |             safeApiCall { | ||||||
|  |                 when { | ||||||
|  |                     it.startsWith("https://animeku.org") -> invokeKuroSource(it, callback) | ||||||
|  |                     else -> loadExtractor(it, mainUrl, subtitleCallback, callback) | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return true | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | @ -0,0 +1,14 @@ | ||||||
|  | 
 | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.lagradost.cloudstream3.plugins.CloudstreamPlugin | ||||||
|  | import com.lagradost.cloudstream3.plugins.Plugin | ||||||
|  | import android.content.Context | ||||||
|  | 
 | ||||||
|  | @CloudstreamPlugin | ||||||
|  | class KuronimeProviderPlugin: Plugin() { | ||||||
|  |     override fun load(context: Context) { | ||||||
|  |         // All providers should be added in this manner. Please don't edit the providers list directly. | ||||||
|  |         registerMainAPI(KuronimeProvider()) | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										27
									
								
								LayarKacaProvider/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								LayarKacaProvider/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,27 @@ | ||||||
|  | // use an integer for version numbers | ||||||
|  | version = 2 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | cloudstream { | ||||||
|  |     language = "id" | ||||||
|  |     // All of these properties are optional, you can safely remove them | ||||||
|  | 
 | ||||||
|  |     // description = "Lorem Ipsum" | ||||||
|  |     authors = listOf("Hexated") | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Status int as the following: | ||||||
|  |      * 0: Down | ||||||
|  |      * 1: Ok | ||||||
|  |      * 2: Slow | ||||||
|  |      * 3: Beta only | ||||||
|  |      * */ | ||||||
|  |     status = 1 // will be 3 if unspecified | ||||||
|  |     tvTypes = listOf( | ||||||
|  |         "AsianDrama", | ||||||
|  |         "TvSeries", | ||||||
|  |         "Movie", | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     iconUrl = "https://www.google.com/s2/favicons?domain=lk21.homes&sz=%size%" | ||||||
|  | } | ||||||
							
								
								
									
										2
									
								
								LayarKacaProvider/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								LayarKacaProvider/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,2 @@ | ||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <manifest package="com.hexated"/> | ||||||
|  | @ -0,0 +1,176 @@ | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.lagradost.cloudstream3.* | ||||||
|  | import com.lagradost.cloudstream3.LoadResponse.Companion.addActors | ||||||
|  | import com.lagradost.cloudstream3.LoadResponse.Companion.addTrailer | ||||||
|  | import com.lagradost.cloudstream3.utils.ExtractorLink | ||||||
|  | import com.lagradost.cloudstream3.utils.loadExtractor | ||||||
|  | import org.jsoup.Jsoup | ||||||
|  | import org.jsoup.nodes.Element | ||||||
|  | 
 | ||||||
|  | class LayarKacaProvider : MainAPI() { | ||||||
|  |     override var mainUrl = "https://lk21.homes" | ||||||
|  |     override var name = "LayarKaca" | ||||||
|  |     override val hasMainPage = true | ||||||
|  |     override var lang = "id" | ||||||
|  |     override val hasDownloadSupport = true | ||||||
|  |     override val supportedTypes = setOf( | ||||||
|  |         TvType.Movie, | ||||||
|  |         TvType.TvSeries, | ||||||
|  |         TvType.AsianDrama | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     override val mainPage = mainPageOf( | ||||||
|  |         "$mainUrl/populer/page/" to "Film Terplopuler", | ||||||
|  |         "$mainUrl/latest-series/page/" to "Series Terbaru", | ||||||
|  |         "$mainUrl/series/asian/page/" to "Film Asian Terbaru", | ||||||
|  |         "$mainUrl/latest/page/" to "Film Upload Terbaru", | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     override suspend fun getMainPage( | ||||||
|  |         page: Int, | ||||||
|  |         request: MainPageRequest | ||||||
|  |     ): HomePageResponse { | ||||||
|  |         val document = app.get(request.data + page).document | ||||||
|  |         val home = document.select("article.mega-item").mapNotNull { | ||||||
|  |             it.toSearchResult() | ||||||
|  |         } | ||||||
|  |         return newHomePageResponse(request.name, home) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun Element.toSearchResult(): SearchResponse? { | ||||||
|  |         val title = this.selectFirst("h1.grid-title > a")?.ownText()?.trim() ?: return null | ||||||
|  |         val href = fixUrl(this.selectFirst("h1.grid-title > a")!!.attr("href")) | ||||||
|  |         val posterUrl = fixUrlNull(this.selectFirst(".grid-poster > a > img")?.attr("src")) | ||||||
|  |         val quality = this.select("div.quality").text().trim() | ||||||
|  |         return newMovieSearchResponse(title, href, TvType.Movie) { | ||||||
|  |             this.posterUrl = posterUrl | ||||||
|  |             addQuality(quality) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun search(query: String): List<SearchResponse> { | ||||||
|  |         val link = "$mainUrl/?s=$query" | ||||||
|  |         val document = app.get(link).document | ||||||
|  | 
 | ||||||
|  |         return document.select("div.search-item").map { | ||||||
|  |             val title = it.selectFirst("h2 > a")!!.text().trim() | ||||||
|  |             val href = it.selectFirst("h2 > a")!!.attr("href") | ||||||
|  |             val posterUrl = fixUrl(it.selectFirst("img.img-thumbnail")?.attr("src").toString()) | ||||||
|  |             newTvSeriesSearchResponse(title, href, TvType.TvSeries) { | ||||||
|  |                 this.posterUrl = posterUrl | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun load(url: String): LoadResponse { | ||||||
|  |         val document = app.get(url).document | ||||||
|  | 
 | ||||||
|  |         val title = document.selectFirst("li.last > span[itemprop=name]")?.text()?.trim().toString() | ||||||
|  |         val poster = fixUrl(document.select("img.img-thumbnail").attr("src").toString()) | ||||||
|  |         val tags = document.select("div.content > div:nth-child(5) > h3 > a").map { it.text() } | ||||||
|  | 
 | ||||||
|  |         val year = Regex("\\d, (\\d+)").find( | ||||||
|  |             document.select("div.content > div:nth-child(7) > h3").text().trim() | ||||||
|  |         )?.groupValues?.get(1).toString().toIntOrNull() | ||||||
|  |         val tvType = if (document.select("div.serial-wrapper") | ||||||
|  |                 .isNotEmpty() | ||||||
|  |         ) TvType.TvSeries else TvType.Movie | ||||||
|  |         val description = document.select("div.content > blockquote").text().trim() | ||||||
|  |         val trailer = document.selectFirst("div.action-player li > a.fancybox")?.attr("href") | ||||||
|  |         val rating = | ||||||
|  |             document.selectFirst("div.content > div:nth-child(6) > h3")?.text()?.toRatingInt() | ||||||
|  |         val actors = | ||||||
|  |             document.select("div.col-xs-9.content > div:nth-child(3) > h3 > a").map { it.text() } | ||||||
|  | 
 | ||||||
|  |         val recommendations = document.select("div.row.item-media").map { | ||||||
|  |             val recName = it.selectFirst("h3")?.text()?.trim().toString() | ||||||
|  |             val recHref = it.selectFirst(".content-media > a")!!.attr("href") | ||||||
|  |             val recPosterUrl = | ||||||
|  |                 fixUrl(it.selectFirst(".poster-media > a > img")?.attr("src").toString()) | ||||||
|  |             newTvSeriesSearchResponse(recName, recHref, TvType.TvSeries) { | ||||||
|  |                 this.posterUrl = recPosterUrl | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return if (tvType == TvType.TvSeries) { | ||||||
|  |             val episodes = document.select("div.episode-list > a:matches(\\d+)").map { | ||||||
|  |                 val href = fixUrl(it.attr("href")) | ||||||
|  |                 val episode = it.text().toIntOrNull() | ||||||
|  |                 val season = | ||||||
|  |                     it.attr("href").substringAfter("season-").substringBefore("-").toIntOrNull() | ||||||
|  |                 Episode( | ||||||
|  |                     href, | ||||||
|  |                     "Episode $episode", | ||||||
|  |                     season, | ||||||
|  |                     episode, | ||||||
|  |                 ) | ||||||
|  |             }.reversed() | ||||||
|  |             newTvSeriesLoadResponse(title, url, TvType.TvSeries, episodes) { | ||||||
|  |                 this.posterUrl = poster | ||||||
|  |                 this.year = year | ||||||
|  |                 this.plot = description | ||||||
|  |                 this.tags = tags | ||||||
|  |                 this.rating = rating | ||||||
|  |                 addActors(actors) | ||||||
|  |                 this.recommendations = recommendations | ||||||
|  |                 addTrailer(trailer) | ||||||
|  |             } | ||||||
|  |         } else { | ||||||
|  |             newMovieLoadResponse(title, url, TvType.Movie, url) { | ||||||
|  |                 this.posterUrl = poster | ||||||
|  |                 this.year = year | ||||||
|  |                 this.plot = description | ||||||
|  |                 this.tags = tags | ||||||
|  |                 this.rating = rating | ||||||
|  |                 addActors(actors) | ||||||
|  |                 this.recommendations = recommendations | ||||||
|  |                 addTrailer(trailer) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun loadLinks( | ||||||
|  |         data: String, | ||||||
|  |         isCasting: Boolean, | ||||||
|  |         subtitleCallback: (SubtitleFile) -> Unit, | ||||||
|  |         callback: (ExtractorLink) -> Unit | ||||||
|  |     ): Boolean { | ||||||
|  | 
 | ||||||
|  |         val document = app.get(data).document | ||||||
|  | 
 | ||||||
|  | //        maybe will need this in future | ||||||
|  | //        val sources = if (data.contains("-episode-")) { | ||||||
|  | //            document.select("script").mapNotNull { script -> | ||||||
|  | //                if (script.data().contains("var data =")) { | ||||||
|  | //                    val scriptData = | ||||||
|  | //                        script.toString().substringAfter("var data = '").substringBefore("';") | ||||||
|  | //                    Jsoup.parse(scriptData).select("li").map { | ||||||
|  | //                        fixUrl(it.select("a").attr("href")) | ||||||
|  | //                    } | ||||||
|  | //                } else { | ||||||
|  | //                    null | ||||||
|  | //                } | ||||||
|  | //            }[0] | ||||||
|  | //        } else { | ||||||
|  | //            document.select("ul#loadProviders > li").map { | ||||||
|  | //                fixUrl(it.select("a").attr("href")) | ||||||
|  | //            } | ||||||
|  | //        } | ||||||
|  | 
 | ||||||
|  |         document.select("ul#loadProviders > li").map { | ||||||
|  |             fixUrl(it.select("a").attr("href")) | ||||||
|  |         }.apmap { | ||||||
|  |             val link = if (it.startsWith("https://layarkacaxxi.icu")) { | ||||||
|  |                 it.substringBeforeLast("/") | ||||||
|  |             } else { | ||||||
|  |                 it | ||||||
|  |             } | ||||||
|  |             loadExtractor(link, data, subtitleCallback, callback) | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return true | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | @ -0,0 +1,14 @@ | ||||||
|  | 
 | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.lagradost.cloudstream3.plugins.CloudstreamPlugin | ||||||
|  | import com.lagradost.cloudstream3.plugins.Plugin | ||||||
|  | import android.content.Context | ||||||
|  | 
 | ||||||
|  | @CloudstreamPlugin | ||||||
|  | class LayarKacaProviderPlugin: Plugin() { | ||||||
|  |     override fun load(context: Context) { | ||||||
|  |         // All providers should be added in this manner. Please don't edit the providers list directly. | ||||||
|  |         registerMainAPI(LayarKacaProvider()) | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										27
									
								
								MultiplexProvider/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								MultiplexProvider/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,27 @@ | ||||||
|  | // use an integer for version numbers | ||||||
|  | version = 1 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | cloudstream { | ||||||
|  |     language = "id" | ||||||
|  |     // All of these properties are optional, you can safely remove them | ||||||
|  | 
 | ||||||
|  |     // description = "Lorem Ipsum" | ||||||
|  |     authors = listOf("Hexated") | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Status int as the following: | ||||||
|  |      * 0: Down | ||||||
|  |      * 1: Ok | ||||||
|  |      * 2: Slow | ||||||
|  |      * 3: Beta only | ||||||
|  |      * */ | ||||||
|  |     status = 1 // will be 3 if unspecified | ||||||
|  |     tvTypes = listOf( | ||||||
|  |         "AsianDrama", | ||||||
|  |         "TvSeries", | ||||||
|  |         "Movie", | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     iconUrl = "https://www.google.com/s2/favicons?domain=146.19.24.137&sz=%size%" | ||||||
|  | } | ||||||
							
								
								
									
										2
									
								
								MultiplexProvider/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								MultiplexProvider/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,2 @@ | ||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <manifest package="com.hexated"/> | ||||||
|  | @ -0,0 +1,188 @@ | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.fasterxml.jackson.annotation.JsonProperty | ||||||
|  | import com.lagradost.cloudstream3.* | ||||||
|  | import com.lagradost.cloudstream3.LoadResponse.Companion.addActors | ||||||
|  | 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.getQualityFromName | ||||||
|  | import org.jsoup.nodes.Element | ||||||
|  | 
 | ||||||
|  | class MultiplexProvider : MainAPI() { | ||||||
|  |     override var mainUrl = "https://146.19.24.137" | ||||||
|  |     override var name = "Multiplex" | ||||||
|  |     override val hasMainPage = true | ||||||
|  |     override var lang = "id" | ||||||
|  |     override val hasDownloadSupport = true | ||||||
|  |     override val supportedTypes = setOf( | ||||||
|  |         TvType.Movie, | ||||||
|  |         TvType.TvSeries, | ||||||
|  |         TvType.AsianDrama | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     override val mainPage = mainPageOf( | ||||||
|  |         "$mainUrl/genre/top-popular-movies/page/" to "Top Popolar Movies", | ||||||
|  |         "$mainUrl/genre/series-ongoing/page/" to "Series Ongoing", | ||||||
|  |         "$mainUrl/genre/series-barat/page/" to "Series Barat", | ||||||
|  |         "$mainUrl/genre/series-korea/page/" to "Series Korea", | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     override suspend fun getMainPage( | ||||||
|  |         page: Int, | ||||||
|  |         request: MainPageRequest | ||||||
|  |     ): HomePageResponse { | ||||||
|  |         val document = app.get(request.data + page).document | ||||||
|  |         val home = document.select("article.item").mapNotNull { | ||||||
|  |             it.toSearchResult() | ||||||
|  |         } | ||||||
|  |         return newHomePageResponse(request.name, home) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun Element.toSearchResult(): SearchResponse? { | ||||||
|  |         val title = this.selectFirst("h2.entry-title > a")?.text()?.trim() ?: return null | ||||||
|  |         val href = fixUrl(this.selectFirst("a")!!.attr("href")) | ||||||
|  |         val posterUrl = fixUrlNull(this.selectFirst("a > img")?.attr("data-src")) | ||||||
|  |         val quality = this.select("div.gmr-quality-item > a").text().trim() | ||||||
|  |         return if (quality.isEmpty()) { | ||||||
|  |             val episode = this.select("div.gmr-numbeps > span").text().toIntOrNull() | ||||||
|  |             newAnimeSearchResponse(title, href, TvType.TvSeries) { | ||||||
|  |                 this.posterUrl = posterUrl | ||||||
|  |                 addSub(episode) | ||||||
|  |             } | ||||||
|  |         } else { | ||||||
|  |             newMovieSearchResponse(title, href, TvType.Movie) { | ||||||
|  |                 this.posterUrl = posterUrl | ||||||
|  |                 addQuality(quality) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun Element.toBottomSearchResult(): SearchResponse? { | ||||||
|  |         val title = this.selectFirst("a > span.idmuvi-rp-title")?.text()?.trim() ?: return null | ||||||
|  |         val href = this.selectFirst("a")!!.attr("href") | ||||||
|  |         val posterUrl = fixUrl(this.selectFirst("a > img")?.attr("data-src").toString()) | ||||||
|  |         return newMovieSearchResponse(title, href, TvType.Movie) { | ||||||
|  |             this.posterUrl = posterUrl | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun search(query: String): List<SearchResponse> { | ||||||
|  |         val link = "$mainUrl/?s=$query&post_type[]=post&post_type[]=tv" | ||||||
|  |         val document = app.get(link).document | ||||||
|  |         return document.select("article.item").mapNotNull { | ||||||
|  |             it.toSearchResult() | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun load(url: String): LoadResponse { | ||||||
|  |         val document = app.get(url).document | ||||||
|  | 
 | ||||||
|  |         val title = | ||||||
|  |             document.selectFirst("h1.entry-title")?.text()?.substringBefore("Season")?.trim() | ||||||
|  |                 .toString() | ||||||
|  |         val poster = | ||||||
|  |             fixUrl(document.selectFirst("figure.pull-left > img")?.attr("data-src").toString()) | ||||||
|  |         val tags = document.select("span.gmr-movie-genre:contains(Genre:) > a").map { it.text() } | ||||||
|  | 
 | ||||||
|  |         val year = | ||||||
|  |             document.select("span.gmr-movie-genre:contains(Year:) > a").text().trim().toIntOrNull() | ||||||
|  |         val tvType = if (url.contains("/tv/")) TvType.TvSeries else TvType.Movie | ||||||
|  |         val description = document.selectFirst("div[itemprop=description] > p")?.text()?.trim() | ||||||
|  |         val trailer = document.selectFirst("ul.gmr-player-nav li a.gmr-trailer-popup")?.attr("href") | ||||||
|  |         val rating = | ||||||
|  |             document.selectFirst("div.gmr-meta-rating > span[itemprop=ratingValue]")?.text() | ||||||
|  |                 ?.toRatingInt() | ||||||
|  |         val actors = document.select("div.gmr-moviedata").last()?.select("span[itemprop=actors]") | ||||||
|  |             ?.map { it.select("a").text() } | ||||||
|  | 
 | ||||||
|  |         val recommendations = document.select("div.idmuvi-rp ul li").mapNotNull { | ||||||
|  |             it.toBottomSearchResult() | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return if (tvType == TvType.TvSeries) { | ||||||
|  |             val episodes = document.select("div.gmr-listseries > a").map { | ||||||
|  |                 val href = fixUrl(it.attr("href")) | ||||||
|  |                 val episode = it.text().split(" ").last().toIntOrNull() | ||||||
|  |                 val season = it.text().split(" ").first().substringAfter("S").toIntOrNull() | ||||||
|  |                 Episode( | ||||||
|  |                     href, | ||||||
|  |                     "Episode $episode", | ||||||
|  |                     season, | ||||||
|  |                     episode, | ||||||
|  |                 ) | ||||||
|  |             } | ||||||
|  |             newTvSeriesLoadResponse(title, url, TvType.TvSeries, episodes) { | ||||||
|  |                 this.posterUrl = poster | ||||||
|  |                 this.year = year | ||||||
|  |                 this.plot = description | ||||||
|  |                 this.tags = tags | ||||||
|  |                 this.rating = rating | ||||||
|  |                 addActors(actors) | ||||||
|  |                 this.recommendations = recommendations | ||||||
|  |                 addTrailer(trailer) | ||||||
|  |             } | ||||||
|  |         } else { | ||||||
|  |             newMovieLoadResponse(title, url, TvType.Movie, url) { | ||||||
|  |                 this.posterUrl = poster | ||||||
|  |                 this.year = year | ||||||
|  |                 this.plot = description | ||||||
|  |                 this.tags = tags | ||||||
|  |                 this.rating = rating | ||||||
|  |                 addActors(actors) | ||||||
|  |                 this.recommendations = recommendations | ||||||
|  |                 addTrailer(trailer) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private data class ResponseSource( | ||||||
|  |         @JsonProperty("file") val file: String, | ||||||
|  |         @JsonProperty("type") val type: String?, | ||||||
|  |         @JsonProperty("label") val label: String? | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     override suspend fun loadLinks( | ||||||
|  |         data: String, | ||||||
|  |         isCasting: Boolean, | ||||||
|  |         subtitleCallback: (SubtitleFile) -> Unit, | ||||||
|  |         callback: (ExtractorLink) -> Unit | ||||||
|  |     ): Boolean { | ||||||
|  | 
 | ||||||
|  |         val document = app.get(data).document | ||||||
|  | 
 | ||||||
|  |         val id = document.selectFirst("div#muvipro_player_content_id")!!.attr("data-id") | ||||||
|  |         val server = app.post( | ||||||
|  |             "$mainUrl/wp-admin/admin-ajax.php", | ||||||
|  |             data = mapOf("action" to "muvipro_player_content", "tab" to "player1", "post_id" to id) | ||||||
|  |         ).document.select("iframe").attr("src") | ||||||
|  | 
 | ||||||
|  |         app.get(server, referer = "$mainUrl/").document.select("script").map { script -> | ||||||
|  |             if (script.data().contains("var config = {")) { | ||||||
|  |                 val source = script.data().substringAfter("sources: [").substringBefore("],") | ||||||
|  |                 tryParseJson<List<ResponseSource>>("[$source]")?.map { m3u -> | ||||||
|  |                     val m3uData = app.get(m3u.file, referer = "https://gdriveplayer.link/").text | ||||||
|  |                     val quality = | ||||||
|  |                         Regex("\\d{3,4}\\.m3u8").findAll(m3uData).map { it.value }.toList() | ||||||
|  |                     quality.forEach { | ||||||
|  |                         callback.invoke( | ||||||
|  |                             ExtractorLink( | ||||||
|  |                                 source = name, | ||||||
|  |                                 name = name, | ||||||
|  |                                 url = m3u.file.replace("video.m3u8", it), | ||||||
|  |                                 referer = "https://gdriveplayer.link/", | ||||||
|  |                                 quality = getQualityFromName("${it.replace(".m3u8", "")}p"), | ||||||
|  |                                 isM3u8 = true | ||||||
|  |                             ) | ||||||
|  |                         ) | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return true | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | @ -0,0 +1,14 @@ | ||||||
|  | 
 | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.lagradost.cloudstream3.plugins.CloudstreamPlugin | ||||||
|  | import com.lagradost.cloudstream3.plugins.Plugin | ||||||
|  | import android.content.Context | ||||||
|  | 
 | ||||||
|  | @CloudstreamPlugin | ||||||
|  | class MultiplexProviderPlugin: Plugin() { | ||||||
|  |     override fun load(context: Context) { | ||||||
|  |         // All providers should be added in this manner. Please don't edit the providers list directly. | ||||||
|  |         registerMainAPI(MultiplexProvider()) | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										28
									
								
								NeonimeProvider/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								NeonimeProvider/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,28 @@ | ||||||
|  | // use an integer for version numbers | ||||||
|  | version = 1 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | cloudstream { | ||||||
|  |     language = "id" | ||||||
|  |     // All of these properties are optional, you can safely remove them | ||||||
|  | 
 | ||||||
|  |     // description = "Lorem Ipsum" | ||||||
|  |     authors = listOf("Hexated") | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * 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", | ||||||
|  |         "Movie", | ||||||
|  |         "OVA", | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     iconUrl = "https://www.google.com/s2/favicons?domain=neonime.watch&sz=%size%" | ||||||
|  | } | ||||||
							
								
								
									
										2
									
								
								NeonimeProvider/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								NeonimeProvider/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,2 @@ | ||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <manifest package="com.hexated"/> | ||||||
							
								
								
									
										178
									
								
								NeonimeProvider/src/main/kotlin/com/hexated/NeonimeProvider.kt
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										178
									
								
								NeonimeProvider/src/main/kotlin/com/hexated/NeonimeProvider.kt
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,178 @@ | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.lagradost.cloudstream3.* | ||||||
|  | import com.lagradost.cloudstream3.LoadResponse.Companion.addTrailer | ||||||
|  | import com.lagradost.cloudstream3.utils.ExtractorLink | ||||||
|  | import com.lagradost.cloudstream3.utils.loadExtractor | ||||||
|  | import org.jsoup.nodes.Element | ||||||
|  | import java.util.* | ||||||
|  | 
 | ||||||
|  | class NeonimeProvider : MainAPI() { | ||||||
|  |     override var mainUrl = "https://neonime.watch" | ||||||
|  |     override var name = "Neonime" | ||||||
|  |     override val hasQuickSearch = false | ||||||
|  |     override val hasMainPage = true | ||||||
|  |     override var lang = "id" | ||||||
|  |     override val hasDownloadSupport = true | ||||||
|  | 
 | ||||||
|  |     override val supportedTypes = setOf( | ||||||
|  |         TvType.Anime, | ||||||
|  |         TvType.AnimeMovie, | ||||||
|  |         TvType.OVA | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     companion object { | ||||||
|  |         fun getType(t: String): TvType { | ||||||
|  |             return if (t.contains("OVA") || t.contains("Special")) TvType.OVA | ||||||
|  |             else if (t.contains("Movie")) TvType.AnimeMovie | ||||||
|  |             else TvType.Anime | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         fun getStatus(t: String): ShowStatus { | ||||||
|  |             return when (t) { | ||||||
|  |                 "Ended"  -> ShowStatus.Completed | ||||||
|  |                 "OnGoing" -> ShowStatus.Ongoing | ||||||
|  |                 "Ongoing" -> ShowStatus.Ongoing | ||||||
|  |                 "In Production" -> ShowStatus.Ongoing | ||||||
|  |                 "Returning Series" -> ShowStatus.Ongoing | ||||||
|  |                 else -> ShowStatus.Completed | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override val mainPage = mainPageOf( | ||||||
|  |         "$mainUrl/episode/page/" to "Episode Terbaru", | ||||||
|  |         "$mainUrl/tvshows/page/" to "Anime Terbaru", | ||||||
|  |         "$mainUrl/movies/page/" to "Movie", | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     override suspend fun getMainPage(page: Int, request: MainPageRequest): HomePageResponse { | ||||||
|  |         val document = app.get(request.data + page).document | ||||||
|  |         val home = document.select("tbody tr,div.item").mapNotNull { | ||||||
|  |                 it.toSearchResult() | ||||||
|  |             } | ||||||
|  |         return newHomePageResponse(request.name, home) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun getProperAnimeLink(uri: String): String { | ||||||
|  |         return when { | ||||||
|  |             uri.contains("/episode") -> { | ||||||
|  |                 val title = uri.substringAfter("$mainUrl/episode/").let { tt -> | ||||||
|  |                     val fixTitle = Regex("(.*)-\\d{1,2}x\\d+").find(tt)?.groupValues?.getOrNull(1).toString() | ||||||
|  |                     when { | ||||||
|  |                         !tt.contains("-season") && !tt.contains(Regex("-1x\\d+")) && !tt.contains("one-piece") -> "$fixTitle-season-${Regex("-(\\d{1,2})x\\d+").find(tt)?.groupValues?.getOrNull(1).toString()}" | ||||||
|  |                         tt.contains("-special") -> fixTitle.replace(Regex("-x\\d+"), "") | ||||||
|  |                         !fixTitle.contains("-subtitle-indonesia") -> "$fixTitle-subtitle-indonesia" | ||||||
|  |                         else -> fixTitle | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  | //                title = when { | ||||||
|  | //                    title.contains("youkoso-jitsuryoku") && !title.contains("-season") -> title.replace("-e-", "-e-tv-") | ||||||
|  | //                    else -> title | ||||||
|  | //                } | ||||||
|  | 
 | ||||||
|  |                 "$mainUrl/tvshows/$title" | ||||||
|  |             } | ||||||
|  |             else -> uri | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun Element.toSearchResult(): AnimeSearchResponse? { | ||||||
|  |         val title = this.selectFirst("td.bb a")?.ownText() ?: this.selectFirst("h2")?.text() ?: return null | ||||||
|  |         val href = getProperAnimeLink(fixUrl(this.select("a").attr("href"))) | ||||||
|  |         val posterUrl = fixUrl(this.select("img").attr("data-src")) | ||||||
|  |         val epNum = this.selectFirst("td.bb span")?.text()?.let { eps -> | ||||||
|  |             Regex("Episode\\s?([0-9]+)").find(eps)?.groupValues?.getOrNull(1)?.toIntOrNull() | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return newAnimeSearchResponse(title, href, TvType.Anime) { | ||||||
|  |             this.posterUrl = posterUrl | ||||||
|  |             addSub(epNum) | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun search(query: String): List<SearchResponse> { | ||||||
|  |         val link = "$mainUrl/?s=$query" | ||||||
|  |         val document = app.get(link).document | ||||||
|  | 
 | ||||||
|  |         return document.select("div.item.episode-home").mapNotNull { | ||||||
|  |             val title = it.selectFirst("div.judul-anime > span")!!.text() | ||||||
|  |             val poster = it.select("img").attr("data-src").toString().trim() | ||||||
|  |             val episodes = it.selectFirst("div.fixyear > h2.text-center")!! | ||||||
|  |                 .text().replace(Regex("[^0-9]"), "").trim().toIntOrNull() | ||||||
|  |             val tvType = getType(it.selectFirst("span.calidad2.episode")?.text().toString()) | ||||||
|  |             val href = getProperAnimeLink(fixUrl(it.selectFirst("a")!!.attr("href"))) | ||||||
|  | 
 | ||||||
|  |             newAnimeSearchResponse(title, href, tvType) { | ||||||
|  |                 this.posterUrl = poster | ||||||
|  |                 addSub(episodes) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun load(url: String): LoadResponse { | ||||||
|  |         val document = app.get(url).document | ||||||
|  | 
 | ||||||
|  |             if (url.contains("movie") || url.contains("live-action")) { | ||||||
|  |                 val mTitle = document.selectFirst(".sbox > .data > h1[itemprop = name]")?.text().toString().trim() | ||||||
|  |                 val mTrailer = document.selectFirst("div.youtube_id iframe")?.attr("data-wpfc-original-src")?.substringAfterLast("html#")?.let{ "https://www.youtube.com/embed/$it"} | ||||||
|  | 
 | ||||||
|  |                 return newMovieLoadResponse(name = mTitle, url = url, type = TvType.Movie, dataUrl = url) { | ||||||
|  |                     posterUrl = document.selectFirst(".sbox > .imagen > .fix > img[itemprop = image]")?.attr("data-src") | ||||||
|  |                     year = document.selectFirst("a[href*=release-year]")!!.text().toIntOrNull() | ||||||
|  |                     plot = document.select("div[itemprop = description]").text().trim() | ||||||
|  |                     rating = document.select("span[itemprop = ratingValue]").text().toIntOrNull() | ||||||
|  |                     tags = document.select("p.meta_dd > a").map { it.text() } | ||||||
|  |                     addTrailer(mTrailer) | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             else { | ||||||
|  |                 val title = document.select("h1[itemprop = name]").text().trim() | ||||||
|  |                 val trailer = document.selectFirst("div.youtube_id_tv iframe")?.attr("data-wpfc-original-src")?.substringAfterLast("html#")?.let{ "https://www.youtube.com/embed/$it"} | ||||||
|  | 
 | ||||||
|  |                 val episodes = document.select("ul.episodios > li").mapNotNull { | ||||||
|  |                     val header = it.selectFirst(".episodiotitle > a")?.ownText().toString() | ||||||
|  |                     val name = Regex("(Episode\\s?[0-9]+)").find(header)?.groupValues?.getOrNull(0) ?: header | ||||||
|  |                     val link = fixUrl(it.selectFirst(".episodiotitle > a")!!.attr("href")) | ||||||
|  |                     Episode(link, name) | ||||||
|  |                 }.reversed() | ||||||
|  | 
 | ||||||
|  |                 return newAnimeLoadResponse(title, url, TvType.Anime) { | ||||||
|  |                     engName = title | ||||||
|  |                     posterUrl = document.selectFirst(".imagen > img")?.attr("data-src") | ||||||
|  |                     year = document.select("#info a[href*=\"-year/\"]").text().toIntOrNull() | ||||||
|  |                     addEpisodes(DubStatus.Subbed, episodes) | ||||||
|  |                     showStatus = getStatus(document.select("div.metadatac > span").last()!!.text().trim()) | ||||||
|  |                     plot = document.select("div[itemprop = description] > p").text().trim() | ||||||
|  |                     tags = document.select("#info a[href*=\"-genre/\"]").map { it.text() } | ||||||
|  |                     addTrailer(trailer) | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun loadLinks( | ||||||
|  |         data: String, | ||||||
|  |         isCasting: Boolean, | ||||||
|  |         subtitleCallback: (SubtitleFile) -> Unit, | ||||||
|  |         callback: (ExtractorLink) -> Unit | ||||||
|  |     ): Boolean { | ||||||
|  |         val source = if(data.contains("movie") || data.contains("live-action")) { | ||||||
|  |             app.get(data).document.select("#player2-1 > div[id*=div]").mapNotNull { | ||||||
|  |                 fixUrl(it.select("iframe").attr("data-src")) | ||||||
|  |             } | ||||||
|  |         } else { | ||||||
|  |             app.get(data).document.select(".player2 > .embed2 > div[id*=player]").mapNotNull { | ||||||
|  |                 fixUrl(it.select("iframe").attr("data-src")) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         source.apmap { | ||||||
|  |             loadExtractor(it, data, subtitleCallback, callback) | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return true | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | @ -0,0 +1,14 @@ | ||||||
|  | 
 | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.lagradost.cloudstream3.plugins.CloudstreamPlugin | ||||||
|  | import com.lagradost.cloudstream3.plugins.Plugin | ||||||
|  | import android.content.Context | ||||||
|  | 
 | ||||||
|  | @CloudstreamPlugin | ||||||
|  | class NeonimeProviderPlugin: Plugin() { | ||||||
|  |     override fun load(context: Context) { | ||||||
|  |         // All providers should be added in this manner. Please don't edit the providers list directly. | ||||||
|  |         registerMainAPI(NeonimeProvider()) | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										27
									
								
								NontonAnimeIDProvider/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								NontonAnimeIDProvider/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,27 @@ | ||||||
|  | // use an integer for version numbers | ||||||
|  | version = 3 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | cloudstream { | ||||||
|  |     language = "id" | ||||||
|  |     // All of these properties are optional, you can safely remove them | ||||||
|  | 
 | ||||||
|  |     // description = "Lorem Ipsum" | ||||||
|  |      authors = listOf("Hexated") | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * 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=75.119.159.228&sz=%size%" | ||||||
|  | } | ||||||
							
								
								
									
										2
									
								
								NontonAnimeIDProvider/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								NontonAnimeIDProvider/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,2 @@ | ||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <manifest package="com.hexated"/> | ||||||
|  | @ -0,0 +1,257 @@ | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.fasterxml.jackson.annotation.JsonProperty | ||||||
|  | import com.lagradost.cloudstream3.* | ||||||
|  | import com.lagradost.cloudstream3.LoadResponse.Companion.addTrailer | ||||||
|  | import com.lagradost.cloudstream3.utils.ExtractorLink | ||||||
|  | import com.lagradost.cloudstream3.utils.loadExtractor | ||||||
|  | import org.jsoup.Jsoup | ||||||
|  | import org.jsoup.nodes.Element | ||||||
|  | 
 | ||||||
|  | class NontonAnimeIDProvider : MainAPI() { | ||||||
|  |     override var mainUrl = "https://75.119.159.228" | ||||||
|  |     override var name = "NontonAnimeID" | ||||||
|  |     override val hasQuickSearch = false | ||||||
|  |     override val hasMainPage = true | ||||||
|  |     override var lang = "id" | ||||||
|  |     override val hasDownloadSupport = true | ||||||
|  | 
 | ||||||
|  |     override val supportedTypes = setOf( | ||||||
|  |         TvType.Anime, | ||||||
|  |         TvType.AnimeMovie, | ||||||
|  |         TvType.OVA | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     companion object { | ||||||
|  |         fun getType(t: String): TvType { | ||||||
|  |             return when { | ||||||
|  |                 t.contains("TV") -> TvType.Anime | ||||||
|  |                 t.contains("Movie") -> TvType.AnimeMovie | ||||||
|  |                 else -> TvType.OVA | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         fun getStatus(t: String): ShowStatus { | ||||||
|  |             return when (t) { | ||||||
|  |                 "Finished Airing" -> ShowStatus.Completed | ||||||
|  |                 "Currently Airing" -> ShowStatus.Ongoing | ||||||
|  |                 else -> ShowStatus.Completed | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun getMainPage(page: Int, request : MainPageRequest): HomePageResponse { | ||||||
|  |         val document = app.get(mainUrl).document | ||||||
|  | 
 | ||||||
|  |         val homePageList = ArrayList<HomePageList>() | ||||||
|  | 
 | ||||||
|  |         document.select("section#postbaru").forEach { block -> | ||||||
|  |             val header = block.selectFirst("h2")!!.text().trim() | ||||||
|  |             val animes = block.select("article.animeseries").mapNotNull { | ||||||
|  |                 it.toSearchResult() | ||||||
|  |             } | ||||||
|  |             if (animes.isNotEmpty()) homePageList.add(HomePageList(header, animes)) | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         document.select("aside#sidebar_right > div.side").forEach { block -> | ||||||
|  |             val header = block.selectFirst("h3")!!.ownText().trim() | ||||||
|  |             val animes = block.select("ul li.fullwdth").mapNotNull { | ||||||
|  |                 it.toSearchResultPopular() | ||||||
|  |             } | ||||||
|  |             if (animes.isNotEmpty()) homePageList.add(HomePageList(header, animes)) | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return HomePageResponse(homePageList) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun getProperAnimeLink(uri: String): String { | ||||||
|  |         return if (uri.contains("/anime/")) { | ||||||
|  |             uri | ||||||
|  |         } else { | ||||||
|  |             var title = uri.substringAfter("$mainUrl/") | ||||||
|  |             val fixTitle = Regex("(.*)-episode.*").find(title)?.groupValues?.getOrNull(1).toString() | ||||||
|  |             title = when { | ||||||
|  |                 title.contains("utawarerumono-season-3") -> fixTitle.replace( | ||||||
|  |                     "season-3", | ||||||
|  |                     "futari-no-hakuoro" | ||||||
|  |                 ) | ||||||
|  |                 title.contains("kingdom-season-4") -> fixTitle.replace("season-4", "4th-season") | ||||||
|  |                 title.contains("maou-sama-season-2") -> fixTitle.replace("season-2", "2") | ||||||
|  |                 title.contains("overlord-season-4") -> fixTitle.replace("season-4", "iv") | ||||||
|  |                 title.contains("kyoushitsu-e-season-2") -> fixTitle.replace( | ||||||
|  |                     "kyoushitsu-e-season-2", | ||||||
|  |                     "kyoushitsu-e-tv-2nd-season" | ||||||
|  |                 ) | ||||||
|  |                 title.contains("season-2") -> fixTitle.replace("season-2", "2nd-season") | ||||||
|  |                 title.contains("season-3") -> fixTitle.replace("season-3", "3rd-season") | ||||||
|  |                 title.contains("movie") -> title.substringBefore("-movie") | ||||||
|  |                 else -> fixTitle | ||||||
|  |             } | ||||||
|  |             "$mainUrl/anime/$title" | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun Element.toSearchResult(): AnimeSearchResponse? { | ||||||
|  |         val href = getProperAnimeLink(fixUrl(this.selectFirst("a")!!.attr("href"))) | ||||||
|  |         val title = this.selectFirst("h3.title")?.text() ?: return null | ||||||
|  |         val posterUrl = fixUrl(this.select("img").attr("data-src")) | ||||||
|  | 
 | ||||||
|  |         return newAnimeSearchResponse(title, href, TvType.Anime) { | ||||||
|  |             this.posterUrl = posterUrl | ||||||
|  |             addDubStatus(dubExist = false, subExist = true) | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun Element.toSearchResultPopular(): AnimeSearchResponse? { | ||||||
|  |         val href = getProperAnimeLink(fixUrl(this.selectFirst("a")!!.attr("href"))) | ||||||
|  |         val title = this.selectFirst("h4")?.text()?.trim() ?: return null | ||||||
|  |         val posterUrl = fixUrl(this.select("img").attr("data-src")) | ||||||
|  | 
 | ||||||
|  |         return newAnimeSearchResponse(title, href, TvType.Anime) { | ||||||
|  |             this.posterUrl = posterUrl | ||||||
|  |             addDubStatus(dubExist = false, subExist = true) | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun search(query: String): List<SearchResponse> { | ||||||
|  |         val link = "$mainUrl/?s=$query" | ||||||
|  |         val document = app.get(link).document | ||||||
|  | 
 | ||||||
|  |         return document.select(".result > ul > li").mapNotNull { | ||||||
|  |             val title = it.selectFirst("h2")!!.text().trim() | ||||||
|  |             val poster = it.selectFirst("img")!!.attr("src") | ||||||
|  |             val tvType = getType( | ||||||
|  |                 it.selectFirst(".boxinfores > span.typeseries")!!.text().toString() | ||||||
|  |             ) | ||||||
|  |             val href = fixUrl(it.selectFirst("a")!!.attr("href")) | ||||||
|  | 
 | ||||||
|  |             newAnimeSearchResponse(title, href, tvType) { | ||||||
|  |                 this.posterUrl = poster | ||||||
|  |                 addDubStatus(dubExist = false, subExist = true) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private data class EpResponse( | ||||||
|  |         @JsonProperty("posts") val posts: String?, | ||||||
|  |         @JsonProperty("max_page") val max_page: Int?, | ||||||
|  |         @JsonProperty("found_posts") val found_posts: Int?, | ||||||
|  |         @JsonProperty("content") val content: String | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     override suspend fun load(url: String): LoadResponse { | ||||||
|  |         val document = app.get(url).document | ||||||
|  | 
 | ||||||
|  |         val title = document.selectFirst("h1.entry-title.cs")!!.text().trim() | ||||||
|  |         val poster = document.selectFirst(".poster > img")?.attr("data-src") | ||||||
|  |         val tags = document.select(".tagline > a").map { it.text() } | ||||||
|  | 
 | ||||||
|  |         val year = Regex("\\d, ([0-9]*)").find( | ||||||
|  |             document.select(".bottomtitle > span:nth-child(5)").text() | ||||||
|  |         )?.groupValues?.get(1)?.toIntOrNull() | ||||||
|  |         val status = getStatus( | ||||||
|  |             document.select("span.statusseries").text().trim() | ||||||
|  |         ) | ||||||
|  |         val type = getType(document.select("span.typeseries").text().trim()) | ||||||
|  |         val rating = document.select("span.nilaiseries").text().trim().toIntOrNull() | ||||||
|  |         val description = document.select(".entry-content.seriesdesc > p").text().trim() | ||||||
|  |         val trailer = document.selectFirst("a.trailerbutton")?.attr("href") | ||||||
|  | 
 | ||||||
|  |         val episodes = if (document.select("button.buttfilter").isNotEmpty()) { | ||||||
|  |             val id = document.select("input[name=series_id]").attr("value") | ||||||
|  |             val numEp = | ||||||
|  |                 document.selectFirst(".latestepisode > a")?.text()?.replace(Regex("[^0-9]"), "") | ||||||
|  |                     .toString() | ||||||
|  |             Jsoup.parse( | ||||||
|  |                 app.post( | ||||||
|  |                     url = "$mainUrl/wp-admin/admin-ajax.php", | ||||||
|  |                     data = mapOf( | ||||||
|  |                         "misha_number_of_results" to numEp, | ||||||
|  |                         "misha_order_by" to "date-DESC", | ||||||
|  |                         "action" to "mishafilter", | ||||||
|  |                         "series_id" to id | ||||||
|  |                     ) | ||||||
|  |                 ).parsed<EpResponse>().content | ||||||
|  |             ).select("li").map { | ||||||
|  |                 val name = Regex("(Episode\\s?[0-9]+)").find( | ||||||
|  |                     it.selectFirst("a")?.text().toString() | ||||||
|  |                 )?.groupValues?.getOrNull(0) ?: it.selectFirst("a")?.text() | ||||||
|  |                 val link = fixUrl(it.selectFirst("a")!!.attr("href")) | ||||||
|  |                 Episode(link, name) | ||||||
|  |             }.reversed() | ||||||
|  |         } else { | ||||||
|  |             document.select("ul.misha_posts_wrap2 > li").map { | ||||||
|  |                 val name = Regex("(Episode\\s?[0-9]+)").find( | ||||||
|  |                     it.selectFirst("a")?.text().toString() | ||||||
|  |                 )?.groupValues?.getOrNull(0) ?: it.selectFirst("a")?.text() | ||||||
|  |                 val link = it.select("a").attr("href") | ||||||
|  |                 Episode(link, name) | ||||||
|  |             }.reversed() | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |         val recommendations = document.select(".result > li").mapNotNull { | ||||||
|  |             val epHref = it.selectFirst("a")!!.attr("href") | ||||||
|  |             val epTitle = it.selectFirst("h3")!!.text() | ||||||
|  |             val epPoster = it.select(".top > img").attr("data-src") | ||||||
|  | 
 | ||||||
|  |             newAnimeSearchResponse(epTitle, epHref, TvType.Anime) { | ||||||
|  |                 this.posterUrl = epPoster | ||||||
|  |                 addDubStatus(dubExist = false, subExist = true) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return newAnimeLoadResponse(title, url, type) { | ||||||
|  |             engName = title | ||||||
|  |             posterUrl = poster | ||||||
|  |             this.year = year | ||||||
|  |             addEpisodes(DubStatus.Subbed, episodes) | ||||||
|  |             showStatus = status | ||||||
|  |             this.rating = rating | ||||||
|  |             plot = description | ||||||
|  |             addTrailer(trailer) | ||||||
|  |             this.tags = tags | ||||||
|  |             this.recommendations = recommendations | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun loadLinks( | ||||||
|  |         data: String, | ||||||
|  |         isCasting: Boolean, | ||||||
|  |         subtitleCallback: (SubtitleFile) -> Unit, | ||||||
|  |         callback: (ExtractorLink) -> Unit | ||||||
|  |     ): Boolean { | ||||||
|  | 
 | ||||||
|  |         val document = app.get(data).document | ||||||
|  |         val sources = ArrayList<String>() | ||||||
|  | 
 | ||||||
|  |         document.select(".container1 > ul > li:not(.boxtab)").apmap { | ||||||
|  |             val dataPost = it.attr("data-post") | ||||||
|  |             val dataNume = it.attr("data-nume") | ||||||
|  |             val dataType = it.attr("data-type") | ||||||
|  | 
 | ||||||
|  |             val iframe = app.post( | ||||||
|  |                 url = "$mainUrl/wp-admin/admin-ajax.php", | ||||||
|  |                 data = mapOf( | ||||||
|  |                     "action" to "player_ajax", | ||||||
|  |                     "post" to dataPost, | ||||||
|  |                     "nume" to dataNume, | ||||||
|  |                     "type" to dataType | ||||||
|  |                 ), | ||||||
|  |                 referer = data, | ||||||
|  |                 headers = mapOf("X-Requested-With" to "XMLHttpRequest") | ||||||
|  |             ).document.select("iframe").attr("src") | ||||||
|  | 
 | ||||||
|  |             sources.add(fixUrl(iframe)) | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         sources.apmap { | ||||||
|  |             loadExtractor(it, "$mainUrl/", subtitleCallback, callback) | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return true | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -0,0 +1,14 @@ | ||||||
|  | 
 | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.lagradost.cloudstream3.plugins.CloudstreamPlugin | ||||||
|  | import com.lagradost.cloudstream3.plugins.Plugin | ||||||
|  | import android.content.Context | ||||||
|  | 
 | ||||||
|  | @CloudstreamPlugin | ||||||
|  | class NontonAnimeIDProviderPlugin: Plugin() { | ||||||
|  |     override fun load(context: Context) { | ||||||
|  |         // All providers should be added in this manner. Please don't edit the providers list directly. | ||||||
|  |         registerMainAPI(NontonAnimeIDProvider()) | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										27
									
								
								OploverzProvider/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								OploverzProvider/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,27 @@ | ||||||
|  | // use an integer for version numbers | ||||||
|  | version = 1 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | cloudstream { | ||||||
|  |     language = "id" | ||||||
|  |     // All of these properties are optional, you can safely remove them | ||||||
|  | 
 | ||||||
|  |     // description = "Lorem Ipsum" | ||||||
|  |     authors = listOf("Hexated") | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * 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=65.108.132.145&sz=%size%" | ||||||
|  | } | ||||||
							
								
								
									
										2
									
								
								OploverzProvider/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								OploverzProvider/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,2 @@ | ||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <manifest package="com.hexated"/> | ||||||
							
								
								
									
										203
									
								
								OploverzProvider/src/main/kotlin/com/hexated/OploverzProvider.kt
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										203
									
								
								OploverzProvider/src/main/kotlin/com/hexated/OploverzProvider.kt
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,203 @@ | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.fasterxml.jackson.annotation.JsonProperty | ||||||
|  | import com.lagradost.cloudstream3.* | ||||||
|  | import com.lagradost.cloudstream3.LoadResponse.Companion.addTrailer | ||||||
|  | import com.lagradost.cloudstream3.utils.* | ||||||
|  | import org.jsoup.Jsoup | ||||||
|  | import org.jsoup.nodes.Element | ||||||
|  | import java.util.ArrayList | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | class OploverzProvider : MainAPI() { | ||||||
|  |     override var mainUrl = "https://65.108.132.145" | ||||||
|  |     override var name = "Oploverz" | ||||||
|  |     override val hasMainPage = true | ||||||
|  |     override var lang = "id" | ||||||
|  |     override val hasDownloadSupport = true | ||||||
|  | 
 | ||||||
|  |     override val supportedTypes = setOf( | ||||||
|  |         TvType.Anime, | ||||||
|  |         TvType.AnimeMovie, | ||||||
|  |         TvType.OVA | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     companion object { | ||||||
|  |         fun getType(t: String): TvType { | ||||||
|  |             return when { | ||||||
|  |                 t.contains("TV") -> TvType.Anime | ||||||
|  |                 t.contains("Movie") -> TvType.AnimeMovie | ||||||
|  |                 else -> TvType.OVA | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         fun getStatus(t: String): ShowStatus { | ||||||
|  |             return when (t) { | ||||||
|  |                 "Completed" -> ShowStatus.Completed | ||||||
|  |                 "Ongoing" -> ShowStatus.Ongoing | ||||||
|  |                 else -> ShowStatus.Completed | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override val mainPage = mainPageOf( | ||||||
|  |         "&status=&type=&order=update" to "Episode Terbaru", | ||||||
|  |         "&status=&type=&order=latest" to "Anime Terbaru", | ||||||
|  |         "&sub=&order=popular" to "Popular Anime", | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     override suspend fun getMainPage( | ||||||
|  |         page: Int, | ||||||
|  |         request: MainPageRequest | ||||||
|  |     ): HomePageResponse { | ||||||
|  |         val document = app.get("$mainUrl/anime/?page=$page${request.data}").document | ||||||
|  |         val home = document.select("article[itemscope=itemscope]").mapNotNull { | ||||||
|  |             it.toSearchResult() | ||||||
|  |         } | ||||||
|  |         return newHomePageResponse(request.name, home) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun getProperAnimeLink(uri: String): String { | ||||||
|  | 
 | ||||||
|  |         return if (uri.contains("/anime/")) { | ||||||
|  |             uri | ||||||
|  |         } else { | ||||||
|  |             var title = uri.substringAfter("$mainUrl/") | ||||||
|  |             title = when { | ||||||
|  |                 (title.contains("-episode")) && !(title.contains("-ova")) -> Regex("(.+)-episode").find( | ||||||
|  |                     title | ||||||
|  |                 )?.groupValues?.get(1).toString() | ||||||
|  |                 (title.contains("-ova")) -> Regex("(.+)-ova").find(title)?.groupValues?.get(1) | ||||||
|  |                     .toString() | ||||||
|  |                 (title.contains("-movie")) -> Regex("(.+)-subtitle").find(title)?.groupValues?.get(1) | ||||||
|  |                     .toString() | ||||||
|  |                 else -> Regex("(.+)-subtitle").find(title)?.groupValues?.get(1).toString() | ||||||
|  |                     .replace(Regex("-\\d+"), "") | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             when { | ||||||
|  |                 title.contains("overlord") -> { | ||||||
|  |                     title = title.replace("s", "season-") | ||||||
|  |                 } | ||||||
|  |                 title.contains("kaguya-sama") -> { | ||||||
|  |                     title = title.replace("s3", "ultra-romantic") | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             "$mainUrl/anime/$title" | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun Element.toSearchResult(): AnimeSearchResponse? { | ||||||
|  |         val href = getProperAnimeLink(this.selectFirst("a.tip")!!.attr("href")) | ||||||
|  |         val title = this.selectFirst("h2[itemprop=headline]")?.text()?.trim() ?: return null | ||||||
|  |         val posterUrl = fixUrlNull(this.selectFirst("img")?.attr("src")) | ||||||
|  |         val type = getType(this.selectFirst(".eggtype, .typez")?.text()?.trim().toString()) | ||||||
|  | 
 | ||||||
|  |         return newAnimeSearchResponse(title, href, type) { | ||||||
|  |             this.posterUrl = posterUrl | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun search(query: String): List<SearchResponse> { | ||||||
|  |         val link = "$mainUrl/?s=$query" | ||||||
|  |         val document = app.get(link).document | ||||||
|  | 
 | ||||||
|  |         return document.select("article[itemscope=itemscope]").map { | ||||||
|  |             val title = it.selectFirst(".tt")?.ownText()?.trim().toString() | ||||||
|  |             val poster = fixUrlNull(it.selectFirst("img")?.attr("src")) | ||||||
|  |             val tvType = getType(it.selectFirst(".typez")?.text().toString()) | ||||||
|  |             val href = fixUrl(it.selectFirst("a.tip")!!.attr("href")) | ||||||
|  | 
 | ||||||
|  |             newAnimeSearchResponse(title, href, tvType) { | ||||||
|  |                 this.posterUrl = poster | ||||||
|  |                 addDubStatus(dubExist = false, subExist = true) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun load(url: String): LoadResponse { | ||||||
|  |         val document = app.get(url).document | ||||||
|  | 
 | ||||||
|  |         val title = document.selectFirst("h1.entry-title")!!.text().trim() | ||||||
|  |         val poster = document.select(".thumb > img").attr("src") | ||||||
|  |         val tags = document.select(".genxed > a").map { it.text() } | ||||||
|  | 
 | ||||||
|  |         val year = Regex("\\d, ([0-9]*)").find( | ||||||
|  |             document.selectFirst(".info-content > .spe > span > time")!!.text().trim() | ||||||
|  |         )?.groupValues?.get(1).toString().toIntOrNull() | ||||||
|  |         val status = getStatus( | ||||||
|  |             document.select(".info-content > .spe > span:nth-child(1)") | ||||||
|  |                 .text().trim().replace("Status: ", "") | ||||||
|  |         ) | ||||||
|  |         val typeCheck = | ||||||
|  |             when (document.select(".info-content > .spe > span:nth-child(5), .info-content > .spe > span") | ||||||
|  |                 .text().trim()) { | ||||||
|  |                 "OVA" -> "OVA" | ||||||
|  |                 "Movie" -> "Movie" | ||||||
|  |                 else -> "TV" | ||||||
|  |             } | ||||||
|  |         val type = getType(typeCheck) | ||||||
|  |         val description = document.select(".entry-content > p").text().trim() | ||||||
|  |         val trailer = document.selectFirst("a.trailerbutton")?.attr("href") | ||||||
|  | 
 | ||||||
|  |         val episodes = document.select(".eplister > ul > li").map { | ||||||
|  |             val header = it.select(".epl-title").text() | ||||||
|  |             val name = | ||||||
|  |                 Regex("(Episode\\s?[0-9]+)").find(header)?.groupValues?.getOrNull(0) ?: header | ||||||
|  |             val link = fixUrl(it.select("a").attr("href")) | ||||||
|  |             Episode(link, name) | ||||||
|  |         }.reversed() | ||||||
|  | 
 | ||||||
|  |         val recommendations = | ||||||
|  |             document.select(".listupd > article[itemscope=itemscope]").mapNotNull { rec -> | ||||||
|  |                 val epTitle = rec.selectFirst(".tt")!!.ownText().trim() | ||||||
|  |                 val epPoster = rec.selectFirst("img")!!.attr("src") | ||||||
|  |                 val epType = getType(rec.selectFirst(".typez")?.text().toString()) | ||||||
|  |                 val epHref = fixUrl(rec.selectFirst("a.tip")!!.attr("href")) | ||||||
|  | 
 | ||||||
|  |                 newAnimeSearchResponse(epTitle, epHref, epType) { | ||||||
|  |                     this.posterUrl = epPoster | ||||||
|  |                     addDubStatus(dubExist = false, subExist = true) | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |         return newAnimeLoadResponse(title, url, type) { | ||||||
|  |             engName = title | ||||||
|  |             posterUrl = poster | ||||||
|  |             this.year = year | ||||||
|  |             addEpisodes(DubStatus.Subbed, episodes) | ||||||
|  |             showStatus = status | ||||||
|  |             plot = description | ||||||
|  |             this.tags = tags | ||||||
|  |             this.recommendations = recommendations | ||||||
|  |             addTrailer(trailer) | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     data class Source( | ||||||
|  |         @JsonProperty("play_url") val play_url: String, | ||||||
|  |         @JsonProperty("format_id") val format_id: Int | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     override suspend fun loadLinks( | ||||||
|  |         data: String, | ||||||
|  |         isCasting: Boolean, | ||||||
|  |         subtitleCallback: (SubtitleFile) -> Unit, | ||||||
|  |         callback: (ExtractorLink) -> Unit | ||||||
|  |     ): Boolean { | ||||||
|  |         val document = app.get(data).document | ||||||
|  |         val sources = document.select(".mobius > .mirror > option").mapNotNull { | ||||||
|  |             fixUrl(Jsoup.parse(base64Decode(it.attr("value"))).select("iframe").attr("src")) | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         sources.apmap { | ||||||
|  |             loadExtractor(it, data, subtitleCallback, callback) | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return true | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | @ -0,0 +1,14 @@ | ||||||
|  | 
 | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.lagradost.cloudstream3.plugins.CloudstreamPlugin | ||||||
|  | import com.lagradost.cloudstream3.plugins.Plugin | ||||||
|  | import android.content.Context | ||||||
|  | 
 | ||||||
|  | @CloudstreamPlugin | ||||||
|  | class OploverzProviderPlugin: Plugin() { | ||||||
|  |     override fun load(context: Context) { | ||||||
|  |         // All providers should be added in this manner. Please don't edit the providers list directly. | ||||||
|  |         registerMainAPI(OploverzProvider()) | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										27
									
								
								OtakudesuProvider/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								OtakudesuProvider/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,27 @@ | ||||||
|  | // use an integer for version numbers | ||||||
|  | version = 1 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | cloudstream { | ||||||
|  |     language = "id" | ||||||
|  |     // All of these properties are optional, you can safely remove them | ||||||
|  | 
 | ||||||
|  |     // description = "Lorem Ipsum" | ||||||
|  |     authors = listOf("Hexated") | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * 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=otakudesu.watch&sz=%size%" | ||||||
|  | } | ||||||
							
								
								
									
										2
									
								
								OtakudesuProvider/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								OtakudesuProvider/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,2 @@ | ||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <manifest package="com.hexated"/> | ||||||
|  | @ -0,0 +1,204 @@ | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.fasterxml.jackson.annotation.JsonProperty | ||||||
|  | import com.lagradost.cloudstream3.* | ||||||
|  | import com.lagradost.cloudstream3.utils.AppUtils.tryParseJson | ||||||
|  | import com.lagradost.cloudstream3.utils.ExtractorLink | ||||||
|  | import com.lagradost.cloudstream3.utils.loadExtractor | ||||||
|  | import org.jsoup.Jsoup | ||||||
|  | import org.jsoup.nodes.Element | ||||||
|  | import java.util.ArrayList | ||||||
|  | 
 | ||||||
|  | class OtakudesuProvider : MainAPI() { | ||||||
|  |     override var mainUrl = "https://otakudesu.watch" | ||||||
|  |     override var name = "Otakudesu" | ||||||
|  |     override val hasMainPage = true | ||||||
|  |     override var lang = "id" | ||||||
|  |     override val hasDownloadSupport = true | ||||||
|  | 
 | ||||||
|  |     override val supportedTypes = setOf( | ||||||
|  |         TvType.Anime, | ||||||
|  |         TvType.AnimeMovie, | ||||||
|  |         TvType.OVA | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     companion object { | ||||||
|  |         fun getType(t: String): TvType { | ||||||
|  |             return if (t.contains("OVA") || t.contains("Special")) TvType.OVA | ||||||
|  |             else if (t.contains("Movie")) TvType.AnimeMovie | ||||||
|  |             else TvType.Anime | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         fun getStatus(t: String): ShowStatus { | ||||||
|  |             return when (t) { | ||||||
|  |                 "Completed" -> ShowStatus.Completed | ||||||
|  |                 "Ongoing" -> ShowStatus.Ongoing | ||||||
|  |                 else -> ShowStatus.Completed | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override val mainPage = mainPageOf( | ||||||
|  |         "$mainUrl/ongoing-anime/page/" to "Anime Ongoing", | ||||||
|  |         "$mainUrl/complete-anime/page/" to "Anime Completed" | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     override suspend fun getMainPage( | ||||||
|  |         page: Int, | ||||||
|  |         request: MainPageRequest | ||||||
|  |     ): HomePageResponse { | ||||||
|  |         val document = app.get(request.data + page).document | ||||||
|  |         val home = document.select("div.venz > ul > li").mapNotNull { | ||||||
|  |             it.toSearchResult() | ||||||
|  |         } | ||||||
|  |         return newHomePageResponse(request.name, home) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun Element.toSearchResult(): AnimeSearchResponse? { | ||||||
|  |         val title = this.selectFirst("h2.jdlflm")?.text()?.trim() ?: return null | ||||||
|  |         val href = this.selectFirst("a")!!.attr("href") | ||||||
|  |         val posterUrl = this.select("div.thumbz > img").attr("src").toString() | ||||||
|  |         val epNum = this.selectFirst("div.epz")?.ownText()?.replace(Regex("[^0-9]"), "")?.trim() | ||||||
|  |             ?.toIntOrNull() | ||||||
|  |         return newAnimeSearchResponse(title, href, TvType.Anime) { | ||||||
|  |             this.posterUrl = posterUrl | ||||||
|  |             addSub(epNum) | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun search(query: String): List<SearchResponse> { | ||||||
|  |         val link = "$mainUrl/?s=$query&post_type=anime" | ||||||
|  |         val document = app.get(link).document | ||||||
|  | 
 | ||||||
|  |         return document.select("ul.chivsrc > li").map { | ||||||
|  |             val title = it.selectFirst("h2 > a")!!.ownText().trim() | ||||||
|  |             val href = it.selectFirst("h2 > a")!!.attr("href") | ||||||
|  |             val posterUrl = it.selectFirst("img")!!.attr("src").toString() | ||||||
|  |             newAnimeSearchResponse(title, href, TvType.Anime) { | ||||||
|  |                 this.posterUrl = posterUrl | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     override suspend fun load(url: String): LoadResponse { | ||||||
|  |         val document = app.get(url).document | ||||||
|  | 
 | ||||||
|  |         val title = document.selectFirst("div.infozingle > p:nth-child(1) > span")?.ownText() | ||||||
|  |             ?.replace(":", "")?.trim().toString() | ||||||
|  |         val poster = document.selectFirst("div.fotoanime > img")?.attr("src") | ||||||
|  |         val tags = document.select("div.infozingle > p:nth-child(11) > span > a").map { it.text() } | ||||||
|  |         val type = getType( | ||||||
|  |             document.selectFirst("div.infozingle > p:nth-child(5) > span")?.ownText() | ||||||
|  |                 ?.replace(":", "")?.trim().toString() | ||||||
|  |         ) | ||||||
|  |         val year = Regex("\\d, ([0-9]*)").find( | ||||||
|  |             document.select("div.infozingle > p:nth-child(9) > span").text() | ||||||
|  |         )?.groupValues?.get(1)?.toIntOrNull() | ||||||
|  |         val status = getStatus( | ||||||
|  |             document.selectFirst("div.infozingle > p:nth-child(6) > span")!!.ownText() | ||||||
|  |                 .replace(":", "") | ||||||
|  |                 .trim() | ||||||
|  |         ) | ||||||
|  |         val description = document.select("div.sinopc > p").text() | ||||||
|  | 
 | ||||||
|  |         val episodes = document.select("div.episodelist")[1].select("ul > li").mapNotNull { | ||||||
|  |             val name = Regex("(Episode\\s?[0-9]+)").find( | ||||||
|  |                 it.selectFirst("a")?.text().toString() | ||||||
|  |             )?.groupValues?.getOrNull(0) ?: it.selectFirst("a")?.text() | ||||||
|  |             val link = fixUrl(it.selectFirst("a")!!.attr("href")) | ||||||
|  |             Episode(link, name) | ||||||
|  |         }.reversed() | ||||||
|  | 
 | ||||||
|  |         val recommendations = | ||||||
|  |             document.select("div.isi-recommend-anime-series > div.isi-konten").map { | ||||||
|  |                 val recName = it.selectFirst("span.judul-anime > a")!!.text() | ||||||
|  |                 val recHref = it.selectFirst("a")!!.attr("href") | ||||||
|  |                 val recPosterUrl = it.selectFirst("a > img")?.attr("src").toString() | ||||||
|  |                 newAnimeSearchResponse(recName, recHref, TvType.Anime) { | ||||||
|  |                     this.posterUrl = recPosterUrl | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |         return newAnimeLoadResponse(title, url, type) { | ||||||
|  |             engName = title | ||||||
|  |             posterUrl = poster | ||||||
|  |             this.year = year | ||||||
|  |             addEpisodes(DubStatus.Subbed, episodes) | ||||||
|  |             showStatus = status | ||||||
|  |             plot = description | ||||||
|  |             this.tags = tags | ||||||
|  |             this.recommendations = recommendations | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     data class ResponseSources( | ||||||
|  |         @JsonProperty("id") val id: String, | ||||||
|  |         @JsonProperty("i") val i: String, | ||||||
|  |         @JsonProperty("q") val q: String, | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     data class ResponseData( | ||||||
|  |         @JsonProperty("data") val data: String | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     override suspend fun loadLinks( | ||||||
|  |         data: String, | ||||||
|  |         isCasting: Boolean, | ||||||
|  |         subtitleCallback: (SubtitleFile) -> Unit, | ||||||
|  |         callback: (ExtractorLink) -> Unit | ||||||
|  |     ): Boolean { | ||||||
|  | 
 | ||||||
|  |         val document = app.get(data).document | ||||||
|  |         val scriptData = document.select("script").last()?.data() | ||||||
|  |         val token = scriptData?.substringAfter("{action:\"")?.substringBefore("\"}").toString() | ||||||
|  | 
 | ||||||
|  |         val nonce = app.post("$mainUrl/wp-admin/admin-ajax.php", data = mapOf("action" to token)) | ||||||
|  |             .parsed<ResponseData>().data | ||||||
|  |         val action = scriptData?.substringAfter(",action:\"")?.substringBefore("\"}").toString() | ||||||
|  | 
 | ||||||
|  |         val mirrorData = document.select("div.mirrorstream > ul > li").mapNotNull { | ||||||
|  |             base64Decode(it.select("a").attr("data-content")) | ||||||
|  |         }.toString() | ||||||
|  | 
 | ||||||
|  |         tryParseJson<List<ResponseSources>>(mirrorData)?.apmap { res -> | ||||||
|  |             val id = res.id | ||||||
|  |             val i = res.i | ||||||
|  |             val q = res.q | ||||||
|  | 
 | ||||||
|  |             var sources = Jsoup.parse( | ||||||
|  |                 base64Decode( | ||||||
|  |                     app.post( | ||||||
|  |                         "${mainUrl}/wp-admin/admin-ajax.php", data = mapOf( | ||||||
|  |                             "id" to id, | ||||||
|  |                             "i" to i, | ||||||
|  |                             "q" to q, | ||||||
|  |                             "nonce" to nonce, | ||||||
|  |                             "action" to action | ||||||
|  |                         ) | ||||||
|  |                     ).parsed<ResponseData>().data | ||||||
|  |                 ) | ||||||
|  |             ).select("iframe").attr("src") | ||||||
|  | 
 | ||||||
|  |             if (sources.startsWith("https://desustream.me")) { | ||||||
|  |                 if (!sources.contains("/arcg/") && !sources.contains("/odchan/") && !sources.contains( | ||||||
|  |                         "/desudrive/" | ||||||
|  |                     ) | ||||||
|  |                 ) { | ||||||
|  |                     sources = app.get(sources).document.select("iframe").attr("src") | ||||||
|  |                 } | ||||||
|  |                 if (sources.startsWith("https://yourupload.com")) { | ||||||
|  |                     sources = sources.replace("//", "//www.") | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             loadExtractor(sources, data, subtitleCallback, callback) | ||||||
|  | 
 | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return true | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | @ -0,0 +1,14 @@ | ||||||
|  | 
 | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.lagradost.cloudstream3.plugins.CloudstreamPlugin | ||||||
|  | import com.lagradost.cloudstream3.plugins.Plugin | ||||||
|  | import android.content.Context | ||||||
|  | 
 | ||||||
|  | @CloudstreamPlugin | ||||||
|  | class OtakudesuProviderPlugin: Plugin() { | ||||||
|  |     override fun load(context: Context) { | ||||||
|  |         // All providers should be added in this manner. Please don't edit the providers list directly. | ||||||
|  |         registerMainAPI(OtakudesuProvider()) | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										28
									
								
								PhimmoichillProvider/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								PhimmoichillProvider/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,28 @@ | ||||||
|  | // use an integer for version numbers | ||||||
|  | version = 2 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | cloudstream { | ||||||
|  |     language = "vi" | ||||||
|  |     // All of these properties are optional, you can safely remove them | ||||||
|  | 
 | ||||||
|  |     // description = "Lorem Ipsum" | ||||||
|  |      authors = listOf("Hexated") | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Status int as the following: | ||||||
|  |      * 0: Down | ||||||
|  |      * 1: Ok | ||||||
|  |      * 2: Slow | ||||||
|  |      * 3: Beta only | ||||||
|  |      * */ | ||||||
|  |     status = 1 // will be 3 if unspecified | ||||||
|  |     tvTypes = listOf( | ||||||
|  |         "AsianDrama", | ||||||
|  |         "Anime", | ||||||
|  |         "TvSeries", | ||||||
|  |         "Movie", | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     iconUrl = "https://www.google.com/s2/favicons?domain=phimmoichill.net&sz=%size%" | ||||||
|  | } | ||||||
							
								
								
									
										2
									
								
								PhimmoichillProvider/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								PhimmoichillProvider/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,2 @@ | ||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <manifest package="com.hexated"/> | ||||||
|  | @ -0,0 +1,192 @@ | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.lagradost.cloudstream3.* | ||||||
|  | import com.lagradost.cloudstream3.LoadResponse.Companion.addActors | ||||||
|  | import com.lagradost.cloudstream3.LoadResponse.Companion.addTrailer | ||||||
|  | import com.lagradost.cloudstream3.mvvm.safeApiCall | ||||||
|  | import com.lagradost.cloudstream3.utils.* | ||||||
|  | import org.jsoup.nodes.Element | ||||||
|  | import java.net.URLDecoder | ||||||
|  | 
 | ||||||
|  | class PhimmoichillProvider : MainAPI() { | ||||||
|  |     override var mainUrl = "https://phimmoichill.net" | ||||||
|  |     override var name = "Phimmoichill" | ||||||
|  |     override val hasMainPage = true | ||||||
|  |     override var lang = "vi" | ||||||
|  |     override val hasDownloadSupport = true | ||||||
|  |     override val supportedTypes = setOf( | ||||||
|  |         TvType.Movie, | ||||||
|  |         TvType.TvSeries, | ||||||
|  |         TvType.Anime, | ||||||
|  |         TvType.AsianDrama | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     override val mainPage = mainPageOf( | ||||||
|  |         "$mainUrl/genre/phim-chieu-rap/page-" to "Phim Chiếu Rạp", | ||||||
|  |         "$mainUrl/list/phim-le/page-" to "Phim Lẻ", | ||||||
|  |         "$mainUrl/list/phim-bo/page-" to "Phim Bộ", | ||||||
|  |         "$mainUrl/genre/phim-hoat-hinh/page-" to "Phim Hoạt Hình", | ||||||
|  |         "$mainUrl/country/phim-han-quoc/page-" to "Phim Hàn Quốc", | ||||||
|  |         "$mainUrl/country/phim-trung-quoc/page-" to "Phim Trung Quốc", | ||||||
|  |         "$mainUrl/country/phim-thai-lan/page-" to "Phim Thái Lan", | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     override suspend fun getMainPage( | ||||||
|  |         page: Int, | ||||||
|  |         request: MainPageRequest | ||||||
|  |     ): HomePageResponse { | ||||||
|  |         val document = app.get(request.data + page).document | ||||||
|  |         val home = document.select("li.item").mapNotNull { | ||||||
|  |             it.toSearchResult() | ||||||
|  |         } | ||||||
|  |         return newHomePageResponse( | ||||||
|  |             list = HomePageList( | ||||||
|  |                 name = request.name, | ||||||
|  |                 list = home, | ||||||
|  |                 isHorizontalImages = true | ||||||
|  |             ), | ||||||
|  |             hasNext = true | ||||||
|  |         ) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun decode(input: String): String? = URLDecoder.decode(input, "utf-8") | ||||||
|  | 
 | ||||||
|  |     private fun Element.toSearchResult(): SearchResponse { | ||||||
|  |         val title = this.selectFirst("p,h3")?.text()?.trim().toString() | ||||||
|  |         val href = fixUrl(this.selectFirst("a")!!.attr("href")) | ||||||
|  |         val posterUrl = decode(this.selectFirst("img")!!.attr("src").substringAfter("url=")) | ||||||
|  |         val temp = this.select("span.label").text() | ||||||
|  |         return if (temp.contains(Regex("\\d"))) { | ||||||
|  |             val episode = Regex("(\\((\\d+))|(\\s(\\d+))").find(temp)?.groupValues?.map { num -> | ||||||
|  |                 num.replace(Regex("\\(|\\s"), "") | ||||||
|  |             }?.distinct()?.firstOrNull()?.toIntOrNull() | ||||||
|  |             newAnimeSearchResponse(title, href, TvType.TvSeries) { | ||||||
|  |                 this.posterUrl = posterUrl | ||||||
|  |                 addSub(episode) | ||||||
|  |             } | ||||||
|  |         } else { | ||||||
|  |             val quality = | ||||||
|  |                 temp.replace(Regex("(-.*)|(\\|.*)|(?i)(VietSub.*)|(?i)(Thuyết.*)"), "").trim() | ||||||
|  |             newMovieSearchResponse(title, href, TvType.Movie) { | ||||||
|  |                 this.posterUrl = posterUrl | ||||||
|  |                 addQuality(quality) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun search(query: String): List<SearchResponse> { | ||||||
|  |         val link = "$mainUrl/tim-kiem/$query" | ||||||
|  |         val document = app.get(link).document | ||||||
|  | 
 | ||||||
|  |         return document.select("ul.list-film li").map { | ||||||
|  |             it.toSearchResult() | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun load(url: String): LoadResponse { | ||||||
|  |         val document = app.get(url).document | ||||||
|  | 
 | ||||||
|  |         val title = document.selectFirst("h1[itemprop=name]")?.text()?.trim().toString() | ||||||
|  |         val link = document.select("ul.list-button li:last-child a").attr("href") | ||||||
|  |         val poster = document.selectFirst("div.image img[itemprop=image]")?.attr("src") | ||||||
|  |         val tags = document.select("ul.entry-meta.block-film li:nth-child(4) a").map { it.text() } | ||||||
|  |         val year = document.select("ul.entry-meta.block-film li:nth-child(2) a").text().trim() | ||||||
|  |             .toIntOrNull() | ||||||
|  |         val tvType = if (document.select("div.latest-episode").isNotEmpty() | ||||||
|  |         ) TvType.TvSeries else TvType.Movie | ||||||
|  |         val description = document.select("div#film-content").text().trim() | ||||||
|  |         val trailer = | ||||||
|  |             document.select("div#trailer script").last()?.data()?.substringAfter("file: \"") | ||||||
|  |                 ?.substringBefore("\",") | ||||||
|  |         val rating = | ||||||
|  |             document.select("ul.entry-meta.block-film li:nth-child(7) span").text().toRatingInt() | ||||||
|  |         val actors = document.select("ul.entry-meta.block-film li:last-child a").map { it.text() } | ||||||
|  |         val recommendations = document.select("ul#list-film-realted li.item").map { | ||||||
|  |             it.toSearchResult().apply { | ||||||
|  |                 this.posterUrl = decode(it.selectFirst("img")!!.attr("data-src").substringAfter("url=")) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return if (tvType == TvType.TvSeries) { | ||||||
|  |             val docEpisodes = app.get(link).document | ||||||
|  |             val episodes = docEpisodes.select("ul#list_episodes > li").map { | ||||||
|  |                 val href = it.select("a").attr("href") | ||||||
|  |                 val episode = | ||||||
|  |                     it.select("a").text().replace(Regex("[^0-9]"), "").trim().toIntOrNull() | ||||||
|  |                 val name = "Episode $episode" | ||||||
|  |                 Episode( | ||||||
|  |                     data = href, | ||||||
|  |                     name = name, | ||||||
|  |                     episode = episode, | ||||||
|  |                 ) | ||||||
|  |             } | ||||||
|  |             newTvSeriesLoadResponse(title, url, TvType.TvSeries, episodes) { | ||||||
|  |                 this.posterUrl = poster | ||||||
|  |                 this.year = year | ||||||
|  |                 this.plot = description | ||||||
|  |                 this.tags = tags | ||||||
|  |                 this.rating = rating | ||||||
|  |                 addActors(actors) | ||||||
|  |                 this.recommendations = recommendations | ||||||
|  |                 addTrailer(trailer) | ||||||
|  |             } | ||||||
|  |         } else { | ||||||
|  |             newMovieLoadResponse(title, url, TvType.Movie, link) { | ||||||
|  |                 this.posterUrl = poster | ||||||
|  |                 this.year = year | ||||||
|  |                 this.plot = description | ||||||
|  |                 this.tags = tags | ||||||
|  |                 this.rating = rating | ||||||
|  |                 addActors(actors) | ||||||
|  |                 this.recommendations = recommendations | ||||||
|  |                 addTrailer(trailer) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun loadLinks( | ||||||
|  |         data: String, | ||||||
|  |         isCasting: Boolean, | ||||||
|  |         subtitleCallback: (SubtitleFile) -> Unit, | ||||||
|  |         callback: (ExtractorLink) -> Unit | ||||||
|  |     ): Boolean { | ||||||
|  |         val document = app.get(data).document | ||||||
|  | 
 | ||||||
|  |         val key = document.select("div#content script") | ||||||
|  |             .find { it.data().contains("filmInfo.episodeID =") }?.data()?.let { script -> | ||||||
|  |                 val id = script.substringAfter("filmInfo.episodeID = parseInt('") | ||||||
|  |                 app.post( | ||||||
|  |                     // Not mainUrl | ||||||
|  |                     url = "https://phimmoichills.net/pmplayer.php", | ||||||
|  |                     data = mapOf("qcao" to id, "sv" to "0"), | ||||||
|  |                     referer = data, | ||||||
|  |                     headers = mapOf( | ||||||
|  |                         "X-Requested-With" to "XMLHttpRequest", | ||||||
|  |                         "Content-Type" to "application/x-www-form-urlencoded; charset=UTF-8" | ||||||
|  |                     ) | ||||||
|  |                 ).text.substringAfterLast("iniPlayers(\"") | ||||||
|  |                     .substringBefore("\",") | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |         listOf( | ||||||
|  |             Pair("https://so-trym.topphimmoi.org/raw/$key/index.m3u8", "PMFAST"), | ||||||
|  |             Pair("https://dash.megacdn.xyz/raw/$key/index.m3u8", "PMHLS"), | ||||||
|  |             Pair("https://dash.megacdn.xyz/dast/$key/index.m3u8", "PMBK") | ||||||
|  |         ).apmap { (link, source) -> | ||||||
|  |             safeApiCall { | ||||||
|  |                 callback.invoke( | ||||||
|  |                     ExtractorLink( | ||||||
|  |                         source, | ||||||
|  |                         source, | ||||||
|  |                         link, | ||||||
|  |                         referer = "$mainUrl/", | ||||||
|  |                         quality = Qualities.P1080.value, | ||||||
|  |                         isM3u8 = true, | ||||||
|  |                     ) | ||||||
|  |                 ) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return true | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | @ -0,0 +1,14 @@ | ||||||
|  | 
 | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.lagradost.cloudstream3.plugins.CloudstreamPlugin | ||||||
|  | import com.lagradost.cloudstream3.plugins.Plugin | ||||||
|  | import android.content.Context | ||||||
|  | 
 | ||||||
|  | @CloudstreamPlugin | ||||||
|  | class PhimmoichillProviderPlugin: Plugin() { | ||||||
|  |     override fun load(context: Context) { | ||||||
|  |         // All providers should be added in this manner. Please don't edit the providers list directly. | ||||||
|  |         registerMainAPI(PhimmoichillProvider()) | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										29
									
								
								RebahinProvider/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								RebahinProvider/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,29 @@ | ||||||
|  | // use an integer for version numbers | ||||||
|  | version = 1 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | cloudstream { | ||||||
|  |     language = "id" | ||||||
|  |     // All of these properties are optional, you can safely remove them | ||||||
|  | 
 | ||||||
|  |     // description = "Lorem Ipsum" | ||||||
|  |     authors = listOf("Hexated") | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Status int as the following: | ||||||
|  |      * 0: Down | ||||||
|  |      * 1: Ok | ||||||
|  |      * 2: Slow | ||||||
|  |      * 3: Beta only | ||||||
|  |      * */ | ||||||
|  |     status = 1 // will be 3 if unspecified | ||||||
|  |     tvTypes = listOf( | ||||||
|  |         "AsianDrama", | ||||||
|  |         "Anime", | ||||||
|  |         "TvSeries", | ||||||
|  |         "Movie", | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     iconUrl = "https://www.google.com/s2/favicons?domain=104.237.198.194&sz=%size%" | ||||||
|  | } | ||||||
							
								
								
									
										2
									
								
								RebahinProvider/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								RebahinProvider/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,2 @@ | ||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <manifest package="com.hexated"/> | ||||||
							
								
								
									
										324
									
								
								RebahinProvider/src/main/kotlin/com/hexated/RebahinProvider.kt
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										324
									
								
								RebahinProvider/src/main/kotlin/com/hexated/RebahinProvider.kt
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,324 @@ | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.fasterxml.jackson.annotation.JsonProperty | ||||||
|  | import com.lagradost.cloudstream3.* | ||||||
|  | import com.lagradost.cloudstream3.LoadResponse.Companion.addActors | ||||||
|  | import com.lagradost.cloudstream3.LoadResponse.Companion.addTrailer | ||||||
|  | import com.lagradost.cloudstream3.mvvm.logError | ||||||
|  | import com.lagradost.cloudstream3.mvvm.safeApiCall | ||||||
|  | import com.lagradost.cloudstream3.network.WebViewResolver | ||||||
|  | import com.lagradost.cloudstream3.utils.AppUtils.tryParseJson | ||||||
|  | import com.lagradost.cloudstream3.utils.ExtractorLink | ||||||
|  | import com.lagradost.cloudstream3.utils.getQualityFromName | ||||||
|  | import com.lagradost.cloudstream3.utils.loadExtractor | ||||||
|  | import org.jsoup.nodes.Element | ||||||
|  | import java.net.URI | ||||||
|  | 
 | ||||||
|  | class RebahinProvider : MainAPI() { | ||||||
|  |     override var mainUrl = "http://104.237.198.194" | ||||||
|  |     override var name = "Rebahin" | ||||||
|  |     override val hasMainPage = true | ||||||
|  |     override var lang = "id" | ||||||
|  |     override val hasDownloadSupport = true | ||||||
|  |     override val supportedTypes = setOf( | ||||||
|  |         TvType.Movie, | ||||||
|  |         TvType.TvSeries, | ||||||
|  |         TvType.Anime, | ||||||
|  |         TvType.AsianDrama | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     override suspend fun getMainPage(page: Int, request : MainPageRequest): HomePageResponse { | ||||||
|  |         val urls = listOf( | ||||||
|  |             Pair("Featured", "xtab1"), | ||||||
|  |             Pair("Film Terbaru", "xtab2"), | ||||||
|  |             Pair("Romance", "xtab3"), | ||||||
|  |             Pair("Drama", "xtab4"), | ||||||
|  |             Pair("Action", "xtab5"), | ||||||
|  |             Pair("Scifi", "xtab6"), | ||||||
|  |             Pair("Tv Series Terbaru", "stab1"), | ||||||
|  |             Pair("Anime Series", "stab2"), | ||||||
|  |             Pair("Drakor Series", "stab3"), | ||||||
|  |             Pair("West Series", "stab4"), | ||||||
|  |             Pair("China Series", "stab5"), | ||||||
|  |             Pair("Japan Series", "stab6"), | ||||||
|  |         ) | ||||||
|  | 
 | ||||||
|  |         val items = ArrayList<HomePageList>() | ||||||
|  | 
 | ||||||
|  |         for ((header, tab) in urls) { | ||||||
|  |             try { | ||||||
|  |                 val home = | ||||||
|  |                     app.get("$mainUrl/wp-content/themes/indoxxi/ajax-top-$tab.php").document.select( | ||||||
|  |                         "div.ml-item" | ||||||
|  |                     ).map { | ||||||
|  |                         it.toSearchResult() | ||||||
|  |                     } | ||||||
|  |                 items.add(HomePageList(header, home)) | ||||||
|  |             } catch (e: Exception) { | ||||||
|  |                 logError(e) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if (items.size <= 0) throw ErrorLoadingException() | ||||||
|  |         return HomePageResponse(items) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun Element.toSearchResult(): SearchResponse { | ||||||
|  |         val title = this.selectFirst("span.mli-info > h2")!!.text().trim() | ||||||
|  |         val href = this.selectFirst("a")!!.attr("href") | ||||||
|  |         val type = | ||||||
|  |             if (this.select("span.mli-quality").isNotEmpty()) TvType.Movie else TvType.TvSeries | ||||||
|  |         return if (type == TvType.Movie) { | ||||||
|  |             val posterUrl = this.select("img").attr("src") | ||||||
|  |             val quality = getQualityFromString(this.select("span.mli-quality").text().trim()) | ||||||
|  |             newMovieSearchResponse(title, href, TvType.Movie) { | ||||||
|  |                 this.posterUrl = posterUrl | ||||||
|  |                 this.quality = quality | ||||||
|  |             } | ||||||
|  |         } else { | ||||||
|  |             val posterUrl = | ||||||
|  |                 this.select("img").attr("src").ifEmpty { this.select("img").attr("data-original") } | ||||||
|  |             val episode = | ||||||
|  |                 this.select("div.mli-eps > span").text().replace(Regex("[^0-9]"), "").toIntOrNull() | ||||||
|  |             newAnimeSearchResponse(title, href, TvType.TvSeries) { | ||||||
|  |                 this.posterUrl = posterUrl | ||||||
|  |                 addSub(episode) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun search(query: String): List<SearchResponse> { | ||||||
|  |         val link = "$mainUrl/?s=$query" | ||||||
|  |         val document = app.get(link).document | ||||||
|  | 
 | ||||||
|  |         return document.select("div.ml-item").map { | ||||||
|  |             it.toSearchResult() | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun load(url: String): LoadResponse { | ||||||
|  |         val document = app.get(url).document | ||||||
|  | 
 | ||||||
|  |         val title = document.selectFirst("h3[itemprop=name]")!!.ownText().trim() | ||||||
|  |         val poster = document.select(".mvic-desc > div.thumb.mvic-thumb").attr("style") | ||||||
|  |             .substringAfter("url(").substringBeforeLast(")") | ||||||
|  |         val tags = document.select("span[itemprop=genre]").map { it.text() } | ||||||
|  | 
 | ||||||
|  |         val year = Regex("([0-9]{4}?)-").find( | ||||||
|  |             document.selectFirst(".mvici-right > p:nth-child(3)")!!.ownText().trim() | ||||||
|  |         )?.groupValues?.get(1).toString().toIntOrNull() | ||||||
|  |         val tvType = if (url.contains("/series/")) TvType.TvSeries else TvType.Movie | ||||||
|  |         val description = document.select("span[itemprop=reviewBody] > p").text().trim() | ||||||
|  |         val trailer = fixUrlNull(document.selectFirst("div.modal-body-trailer iframe")?.attr("src")) | ||||||
|  |         val rating = document.selectFirst("span[itemprop=ratingValue]")?.text()?.toRatingInt() | ||||||
|  |         val duration = document.selectFirst(".mvici-right > p:nth-child(1)")!! | ||||||
|  |             .ownText().replace(Regex("[^0-9]"), "").toIntOrNull() | ||||||
|  |         val actors = document.select("span[itemprop=actor] > a").map { it.select("span").text() } | ||||||
|  | 
 | ||||||
|  |         val baseLink = fixUrl(document.select("div#mv-info > a").attr("href").toString()) | ||||||
|  | 
 | ||||||
|  |         return if (tvType == TvType.TvSeries) { | ||||||
|  |             val episodes = app.get(baseLink).document.select("div#list-eps > a").map { | ||||||
|  |                 Pair(it.text(), it.attr("data-iframe")) | ||||||
|  |             }.groupBy { it.first }.map { eps -> | ||||||
|  |                 Episode( | ||||||
|  |                     data = eps.value.map { fixUrl(base64Decode(it.second)) }.toString(), | ||||||
|  |                     name = eps.key, | ||||||
|  |                     episode = eps.key.filter { it.isDigit() }.toIntOrNull() | ||||||
|  |                 ) | ||||||
|  | 
 | ||||||
|  |             } | ||||||
|  |             newTvSeriesLoadResponse(title, url, TvType.TvSeries, episodes) { | ||||||
|  |                 this.posterUrl = poster | ||||||
|  |                 this.year = year | ||||||
|  |                 this.plot = description | ||||||
|  |                 this.tags = tags | ||||||
|  |                 this.rating = rating | ||||||
|  |                 this.duration = duration | ||||||
|  |                 addActors(actors) | ||||||
|  |                 addTrailer(trailer) | ||||||
|  |             } | ||||||
|  |         } else { | ||||||
|  |             val links = | ||||||
|  |                 app.get(baseLink).document.select("div#server-list div.server-wrapper div[id*=episode]") | ||||||
|  |                     .map { | ||||||
|  |                         fixUrl(base64Decode(it.attr("data-iframe"))) | ||||||
|  |                     }.toString() | ||||||
|  |             newMovieLoadResponse(title, url, TvType.Movie, links) { | ||||||
|  |                 this.posterUrl = poster | ||||||
|  |                 this.year = year | ||||||
|  |                 this.plot = description | ||||||
|  |                 this.tags = tags | ||||||
|  |                 this.rating = rating | ||||||
|  |                 this.duration = duration | ||||||
|  |                 addActors(actors) | ||||||
|  |                 addTrailer(trailer) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private suspend fun invokeLokalSource( | ||||||
|  |         url: String, | ||||||
|  |         name: String, | ||||||
|  |         ref: String, | ||||||
|  |         subCallback: (SubtitleFile) -> Unit, | ||||||
|  |         sourceCallback: (ExtractorLink) -> Unit | ||||||
|  |     ) { | ||||||
|  |         val document = app.get( | ||||||
|  |             url, | ||||||
|  |             allowRedirects = false, | ||||||
|  |             referer = mainUrl, | ||||||
|  |             headers = mapOf("Accept" to "text/html,application/xhtml+xml,application/xml;q=0.9,image/avif,image/webp,*/*;q=0.8") | ||||||
|  |         ).document | ||||||
|  | 
 | ||||||
|  |         document.select("script").map { script -> | ||||||
|  |             if (script.data().contains("sources: [")) { | ||||||
|  |                 val source = tryParseJson<ResponseLocal>( | ||||||
|  |                     script.data().substringAfter("sources: [").substringBefore("],") | ||||||
|  |                 ) | ||||||
|  |                 val m3uData = app.get(source!!.file, referer = ref).text | ||||||
|  |                 val quality = Regex("\\d{3,4}\\.m3u8").findAll(m3uData).map { it.value }.toList() | ||||||
|  | 
 | ||||||
|  |                 quality.forEach { | ||||||
|  |                     sourceCallback.invoke( | ||||||
|  |                         ExtractorLink( | ||||||
|  |                             source = name, | ||||||
|  |                             name = name, | ||||||
|  |                             url = source.file.replace("video.m3u8", it), | ||||||
|  |                             referer = ref, | ||||||
|  |                             quality = getQualityFromName("${it.replace(".m3u8", "")}p"), | ||||||
|  |                             isM3u8 = true | ||||||
|  |                         ) | ||||||
|  |                     ) | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 val trackJson = script.data().substringAfter("tracks: [").substringBefore("],") | ||||||
|  |                 val track = tryParseJson<List<Tracks>>("[$trackJson]") | ||||||
|  |                 track?.map { | ||||||
|  |                     subCallback.invoke( | ||||||
|  |                         SubtitleFile( | ||||||
|  |                             "Indonesian", | ||||||
|  |                             (if (it.file.contains(".srt")) it.file else null)!! | ||||||
|  |                         ) | ||||||
|  |                     ) | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private suspend fun invokeKotakAjairSource( | ||||||
|  |         url: String, | ||||||
|  |         subCallback: (SubtitleFile) -> Unit, | ||||||
|  |         sourceCallback: (ExtractorLink) -> Unit | ||||||
|  |     ) { | ||||||
|  |         val domainUrl = "https://kotakajair.xyz" | ||||||
|  |         val id = url.trimEnd('/').split("/").last() | ||||||
|  |         val sources = app.post( | ||||||
|  |             url = "$domainUrl/api/source/$id", | ||||||
|  |             data = mapOf("r" to mainUrl, "d" to URI(url).host) | ||||||
|  |         ).parsed<ResponseKotakAjair>() | ||||||
|  | 
 | ||||||
|  |         sources.data?.map { | ||||||
|  |             sourceCallback.invoke( | ||||||
|  |                 ExtractorLink( | ||||||
|  |                     name, | ||||||
|  |                     "KotakAjair", | ||||||
|  |                     fixUrl(it.file), | ||||||
|  |                     referer = url, | ||||||
|  |                     quality = getQualityFromName(it.label) | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |         } | ||||||
|  |         val userData = sources.player.poster_file.split("/")[2] | ||||||
|  |         sources.captions?.map { | ||||||
|  |             subCallback.invoke( | ||||||
|  |                 SubtitleFile( | ||||||
|  |                     if (it.language.lowercase().contains("eng")) it.language else "Indonesian", | ||||||
|  |                     "$domainUrl/asset/userdata/$userData/caption/${it.hash}/${it.id}.srt" | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun loadLinks( | ||||||
|  |         data: String, | ||||||
|  |         isCasting: Boolean, | ||||||
|  |         subtitleCallback: (SubtitleFile) -> Unit, | ||||||
|  |         callback: (ExtractorLink) -> Unit | ||||||
|  |     ): Boolean { | ||||||
|  | 
 | ||||||
|  |         data.removeSurrounding("[", "]").split(",").map { it.trim() }.apmap { link -> | ||||||
|  |             safeApiCall { | ||||||
|  |                 when { | ||||||
|  |                     link.startsWith("http://172.96.161.72") -> invokeLokalSource( | ||||||
|  |                         link, | ||||||
|  |                         this.name, | ||||||
|  |                         "http://172.96.161.72/", | ||||||
|  |                         subtitleCallback, | ||||||
|  |                         callback | ||||||
|  |                     ) | ||||||
|  |                     link.startsWith("https://kotakajair.xyz") -> invokeKotakAjairSource( | ||||||
|  |                         link, | ||||||
|  |                         subtitleCallback, | ||||||
|  |                         callback | ||||||
|  |                     ) | ||||||
|  |                     else -> { | ||||||
|  |                         loadExtractor(link, "$mainUrl/", subtitleCallback, callback) | ||||||
|  |                         if (link.startsWith("https://sbfull.com")) { | ||||||
|  |                             val response = app.get( | ||||||
|  |                                 link, interceptor = WebViewResolver( | ||||||
|  |                                     Regex("""\.srt""") | ||||||
|  |                                 ) | ||||||
|  |                             ) | ||||||
|  |                             subtitleCallback.invoke( | ||||||
|  |                                 SubtitleFile( | ||||||
|  |                                     "Indonesian", | ||||||
|  |                                     response.url | ||||||
|  |                                 ) | ||||||
|  |                             ) | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return true | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private data class ResponseLocal( | ||||||
|  |         @JsonProperty("file") val file: String, | ||||||
|  |         @JsonProperty("label") val label: String, | ||||||
|  |         @JsonProperty("type") val type: String? | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     private data class Tracks( | ||||||
|  |         @JsonProperty("file") val file: String, | ||||||
|  |         @JsonProperty("label") val label: String?, | ||||||
|  |         @JsonProperty("kind") val kind: String? | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     private data class Captions( | ||||||
|  |         @JsonProperty("id") val id: String, | ||||||
|  |         @JsonProperty("hash") val hash: String, | ||||||
|  |         @JsonProperty("language") val language: String, | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     private data class Data( | ||||||
|  |         @JsonProperty("file") val file: String, | ||||||
|  |         @JsonProperty("label") val label: String, | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     private data class Player( | ||||||
|  |         @JsonProperty("poster_file") val poster_file: String, | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     private data class ResponseKotakAjair( | ||||||
|  |         @JsonProperty("success") val success: Boolean, | ||||||
|  |         @JsonProperty("player") val player: Player, | ||||||
|  |         @JsonProperty("data") val data: List<Data>?, | ||||||
|  |         @JsonProperty("captions") val captions: List<Captions>? | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | @ -0,0 +1,14 @@ | ||||||
|  | 
 | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.lagradost.cloudstream3.plugins.CloudstreamPlugin | ||||||
|  | import com.lagradost.cloudstream3.plugins.Plugin | ||||||
|  | import android.content.Context | ||||||
|  | 
 | ||||||
|  | @CloudstreamPlugin | ||||||
|  | class RebahinProviderPlugin: Plugin() { | ||||||
|  |     override fun load(context: Context) { | ||||||
|  |         // All providers should be added in this manner. Please don't edit the providers list directly. | ||||||
|  |         registerMainAPI(RebahinProvider()) | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										27
									
								
								TocanimeProvider/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								TocanimeProvider/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,27 @@ | ||||||
|  | // use an integer for version numbers | ||||||
|  | version = 1 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | cloudstream { | ||||||
|  |     language = "vi" | ||||||
|  |     // All of these properties are optional, you can safely remove them | ||||||
|  | 
 | ||||||
|  |     // description = "Lorem Ipsum" | ||||||
|  |     authors = listOf("Hexated") | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * 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=tocanime.co&sz=%size%" | ||||||
|  | } | ||||||
							
								
								
									
										2
									
								
								TocanimeProvider/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								TocanimeProvider/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,2 @@ | ||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <manifest package="com.hexated"/> | ||||||
							
								
								
									
										178
									
								
								TocanimeProvider/src/main/kotlin/com/hexated/TocanimeProvider.kt
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										178
									
								
								TocanimeProvider/src/main/kotlin/com/hexated/TocanimeProvider.kt
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,178 @@ | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.fasterxml.jackson.annotation.JsonProperty | ||||||
|  | import com.lagradost.cloudstream3.* | ||||||
|  | import com.lagradost.cloudstream3.LoadResponse.Companion.addTrailer | ||||||
|  | import com.lagradost.cloudstream3.utils.ExtractorLink | ||||||
|  | import com.lagradost.cloudstream3.utils.Qualities | ||||||
|  | import org.jsoup.nodes.Element | ||||||
|  | import java.util.* | ||||||
|  | 
 | ||||||
|  | class TocanimeProvider : MainAPI() { | ||||||
|  |     override var mainUrl = "https://tocanime.co" | ||||||
|  |     override var name = "Tocanime" | ||||||
|  |     override val hasMainPage = true | ||||||
|  |     override var lang = "vi" | ||||||
|  |     override val hasDownloadSupport = true | ||||||
|  | 
 | ||||||
|  |     override val supportedTypes = setOf( | ||||||
|  |         TvType.Anime, | ||||||
|  |         TvType.AnimeMovie, | ||||||
|  |         TvType.OVA | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     companion object { | ||||||
|  |         fun getType(t: String): TvType { | ||||||
|  |             return when { | ||||||
|  |                 t.contains("OVA") || t.contains("Special") -> TvType.OVA | ||||||
|  |                 t.contains("Movie") -> TvType.AnimeMovie | ||||||
|  |                 else -> TvType.Anime | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         fun getStatus(t: String): ShowStatus { | ||||||
|  |             return when (t) { | ||||||
|  |                 "Đã hoàn thành" -> ShowStatus.Completed | ||||||
|  |                 "Chưa hoàn thành" -> ShowStatus.Ongoing | ||||||
|  |                 else -> ShowStatus.Completed | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun getMainPage(page: Int, request : MainPageRequest): HomePageResponse { | ||||||
|  |         val document = app.get(mainUrl).document | ||||||
|  | 
 | ||||||
|  |         val homePageList = ArrayList<HomePageList>() | ||||||
|  | 
 | ||||||
|  |         document.select("div#playlists > div").forEach { block -> | ||||||
|  |             val header = block.selectFirst("h2")?.text()?.trim() ?: "" | ||||||
|  |             val items = block.select("div.col-lg-3.col-md-4.col-6").map { | ||||||
|  |                 it.toSearchResult() | ||||||
|  |             } | ||||||
|  |             if (items.isNotEmpty()) homePageList.add(HomePageList(header, items)) | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return HomePageResponse(homePageList) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun Element.toSearchResult(): AnimeSearchResponse { | ||||||
|  |         val title = this.selectFirst("h3 a")?.text()?.trim() ?: "" | ||||||
|  |         val href = fixUrl(this.selectFirst("h3 a")!!.attr("href")) | ||||||
|  |         val posterUrl = fixUrlNull(this.selectFirst("div.card-item-img")?.attr("data-original")) | ||||||
|  |         val epNum = this.selectFirst("div.card-item-badget.rtl")?.text()?.let { eps -> | ||||||
|  |             val num = eps.filter { it.isDigit() }.toIntOrNull() | ||||||
|  |             if(eps.contains("Preview")) { | ||||||
|  |                 num?.minus(1) | ||||||
|  |             } else { | ||||||
|  |                 num | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return newAnimeSearchResponse(title, href, TvType.Anime) { | ||||||
|  |             this.posterUrl = posterUrl | ||||||
|  |             addSub(epNum) | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun search(query: String): List<SearchResponse> { | ||||||
|  |         val document = app.get("$mainUrl/content/search?t=kw&q=$query").document | ||||||
|  | 
 | ||||||
|  |         return document.select("div.col-lg-3.col-md-4.col-6").map { | ||||||
|  |             it.toSearchResult() | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun load(url: String): LoadResponse? { | ||||||
|  |         val document = app.get(url).document | ||||||
|  | 
 | ||||||
|  |         val title = document.selectFirst("h1.title")?.text() ?: return null | ||||||
|  |         val type = | ||||||
|  |             if (document.select("div.me-list.scroller a").size == 1) TvType.AnimeMovie else TvType.Anime | ||||||
|  |         val episodes = document.select("div.me-list.scroller a").mapNotNull { | ||||||
|  |             Episode(fixUrl(it.attr("href")), it.text()) | ||||||
|  |         }.reversed() | ||||||
|  |         val trailer = | ||||||
|  |             document.selectFirst("div#trailer script")?.data()?.substringAfter("<iframe src=\"") | ||||||
|  |                 ?.substringBefore("\"") | ||||||
|  | 
 | ||||||
|  |         return newAnimeLoadResponse(title, url, type) { | ||||||
|  |             posterUrl = fixUrlNull(document.selectFirst("img.img")?.attr("data-original")) | ||||||
|  |             year = document.select("dl.movie-des dd")[1].text().split("/").last().toIntOrNull() | ||||||
|  |             showStatus = getStatus( | ||||||
|  |                 document.select("dl.movie-des dd")[0].text() | ||||||
|  |                     .toString() | ||||||
|  |             ) | ||||||
|  |             plot = document.select("div.box-content > p").text() | ||||||
|  |             tags = document.select("dl.movie-des dd")[4].select("li") | ||||||
|  |                 .map { it.select("a").text().removeSuffix(",").trim() } | ||||||
|  |             recommendations = | ||||||
|  |                 document.select("div.col-lg-3.col-md-4.col-6").map { it.toSearchResult() } | ||||||
|  |             addEpisodes(DubStatus.Subbed, episodes) | ||||||
|  |             addTrailer(trailer) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun encode(input: String): String? = java.net.URLEncoder.encode(input, "utf-8") | ||||||
|  | 
 | ||||||
|  |     override suspend fun loadLinks( | ||||||
|  |         data: String, | ||||||
|  |         isCasting: Boolean, | ||||||
|  |         subtitleCallback: (SubtitleFile) -> Unit, | ||||||
|  |         callback: (ExtractorLink) -> Unit | ||||||
|  |     ): Boolean { | ||||||
|  | 
 | ||||||
|  |         val document = app.get( | ||||||
|  |             data, | ||||||
|  |             headers = mapOf("Accept" to "text/html,application/xhtml+xml,application/xml;q=0.9,image/avif,image/webp,*/*;q=0.8"), | ||||||
|  |         ).document | ||||||
|  | 
 | ||||||
|  |         document.select("script").apmap { script -> | ||||||
|  |             if (script.data().contains("var PnPlayer=")) { | ||||||
|  |                 val key = script.data().substringAfter("\"btsurl\":[[").substringBefore("]}]") | ||||||
|  |                     .replace("]", "").replace("\"", "").split(",") | ||||||
|  |                 val id = data.split("_").last().substringBefore(".html") | ||||||
|  | 
 | ||||||
|  |                 app.get( | ||||||
|  |                     url = "$mainUrl/content/parseUrl?v=2&len=0&prefer=&ts=${Date().time}&item_id=$id&username=$id&sv=btsurl&${ | ||||||
|  |                         encode( | ||||||
|  |                             "bts_url[]" | ||||||
|  |                         ) | ||||||
|  |                     }=${ | ||||||
|  |                         encode( | ||||||
|  |                             key.first() | ||||||
|  |                         ) | ||||||
|  |                     }&sig=${key.last()}", | ||||||
|  |                     referer = data, | ||||||
|  |                     headers = mapOf( | ||||||
|  |                         "Accept" to "application/json, text/javascript, */*; q=0.01", | ||||||
|  |                         "X-Requested-With" to "XMLHttpRequest" | ||||||
|  |                     ) | ||||||
|  |                 ).parsedSafe<Responses>()?.let { res -> | ||||||
|  |                     callback.invoke( | ||||||
|  |                         ExtractorLink( | ||||||
|  |                             source = this.name, | ||||||
|  |                             name = this.name, | ||||||
|  |                             url = res.formats?.auto ?: return@let, | ||||||
|  |                             referer = "$mainUrl/", | ||||||
|  |                             quality = Qualities.Unknown.value, | ||||||
|  |                             isM3u8 = true | ||||||
|  |                         ) | ||||||
|  |                     ) | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return true | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     data class Formats( | ||||||
|  |         @JsonProperty("auto") val auto: String?, | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     data class Responses( | ||||||
|  |         @JsonProperty("formats") val formats: Formats?, | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | @ -0,0 +1,14 @@ | ||||||
|  | 
 | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.lagradost.cloudstream3.plugins.CloudstreamPlugin | ||||||
|  | import com.lagradost.cloudstream3.plugins.Plugin | ||||||
|  | import android.content.Context | ||||||
|  | 
 | ||||||
|  | @CloudstreamPlugin | ||||||
|  | class TocanimeProviderPlugin: Plugin() { | ||||||
|  |     override fun load(context: Context) { | ||||||
|  |         // All providers should be added in this manner. Please don't edit the providers list directly. | ||||||
|  |         registerMainAPI(TocanimeProvider()) | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										23
									
								
								Topdocumentaryfilms/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								Topdocumentaryfilms/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,23 @@ | ||||||
|  | // use an integer for version numbers | ||||||
|  | version = 1 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | cloudstream { | ||||||
|  |     language = "en" | ||||||
|  |     // All of these properties are optional, you can safely remove them | ||||||
|  | 
 | ||||||
|  |     // description = "Lorem Ipsum" | ||||||
|  |      authors = listOf("Hexated") | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Status int as the following: | ||||||
|  |      * 0: Down | ||||||
|  |      * 1: Ok | ||||||
|  |      * 2: Slow | ||||||
|  |      * 3: Beta only | ||||||
|  |      * */ | ||||||
|  |     status = 1 // will be 3 if unspecified | ||||||
|  |     tvTypes = listOf("Documentary") | ||||||
|  | 
 | ||||||
|  |     iconUrl = "https://www.google.com/s2/favicons?domain=topdocumentaryfilms.com&sz=%size%" | ||||||
|  | } | ||||||
							
								
								
									
										2
									
								
								Topdocumentaryfilms/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								Topdocumentaryfilms/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,2 @@ | ||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <manifest package="com.hexated"/> | ||||||
|  | @ -0,0 +1,101 @@ | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.lagradost.cloudstream3.* | ||||||
|  | import com.lagradost.cloudstream3.LoadResponse.Companion.addTrailer | ||||||
|  | import com.lagradost.cloudstream3.utils.ExtractorLink | ||||||
|  | import com.lagradost.cloudstream3.utils.loadExtractor | ||||||
|  | import org.jsoup.nodes.Element | ||||||
|  | 
 | ||||||
|  | class Topdocumentaryfilms : MainAPI() { | ||||||
|  |     override var mainUrl = "https://topdocumentaryfilms.com/" | ||||||
|  |     override var name = "Topdocumentaryfilms" | ||||||
|  |     override val hasMainPage = true | ||||||
|  |     override val hasDownloadSupport = true | ||||||
|  |     override val supportedTypes = setOf(TvType.Documentary) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     override val mainPage = mainPageOf( | ||||||
|  |         "$mainUrl/category/technology/page/" to "Technology", | ||||||
|  |         "$mainUrl/category/military-war/page/" to " Military and War", | ||||||
|  |         "$mainUrl/category/sports/page/" to "Sports", | ||||||
|  |         "$mainUrl/category/media/page/" to "Media", | ||||||
|  |         "$mainUrl/category/society/page/" to "Society", | ||||||
|  |         "$mainUrl/category/history/page/" to "History", | ||||||
|  |         "$mainUrl/category/sex/page/" to "Sexuality", | ||||||
|  |         "$mainUrl/category/health/page/" to "Health", | ||||||
|  |         "$mainUrl/category/science-technology/page/" to "Science", | ||||||
|  |         "$mainUrl/category/environment/page/" to "Environment", | ||||||
|  |         "$mainUrl/category/religion/page/" to "Religion", | ||||||
|  |         "$mainUrl/category/economics/page/" to "Economics", | ||||||
|  |         "$mainUrl/category/psychology/page/" to "Psychology", | ||||||
|  |         "$mainUrl/category/drugs/page/" to "Drugs", | ||||||
|  |         "$mainUrl/category/politics/page/" to "Politics", | ||||||
|  |         "$mainUrl/category/crime/page/" to "Crime", | ||||||
|  |         "$mainUrl/category/philosophy/page/" to "Philosophy", | ||||||
|  |         "$mainUrl/category/crime-conspiracy/page/" to "Conspiracy", | ||||||
|  |         "$mainUrl/category/music-performing-arts/page/" to "Performing Arts", | ||||||
|  |         "$mainUrl/category/biography/page/" to "Biography", | ||||||
|  |         "$mainUrl/category/nature-wildlife/page/" to "Nature", | ||||||
|  |         "$mainUrl/category/art-artists/page/" to " Art and Artists", | ||||||
|  |         "$mainUrl/category/mystery/page/" to "Mystery", | ||||||
|  |         "$mainUrl/category/911/page/" to "9/11", | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     override suspend fun getMainPage( | ||||||
|  |         page: Int, | ||||||
|  |         request: MainPageRequest | ||||||
|  |     ): HomePageResponse { | ||||||
|  |         val document = app.get(request.data + page).document | ||||||
|  |         val home = document.select("main article").mapNotNull { | ||||||
|  |             it.toSearchResult() | ||||||
|  |         } | ||||||
|  |         return newHomePageResponse(request.name, home) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun Element.toSearchResult(): SearchResponse? { | ||||||
|  |         val title = this.selectFirst("h2 > a")?.text() ?: return null | ||||||
|  |         val href = this.selectFirst("h2 > a")!!.attr("href") | ||||||
|  |         val posterUrl = this.selectFirst("a img")?.let { | ||||||
|  |             if (it.attr("data-src").isNullOrBlank()) it.attr("src") else it.attr("data-src") | ||||||
|  |         } | ||||||
|  |         return newMovieSearchResponse(title, href, TvType.Documentary) { | ||||||
|  |             this.posterUrl = posterUrl | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun load(url: String): LoadResponse? { | ||||||
|  |         val document = app.get(url).document | ||||||
|  |         val title = document.selectFirst("header > h1")?.text() ?: return null | ||||||
|  |         val link = document.selectFirst("article meta[itemprop=embedUrl]")?.attr("content")?.split("/")?.last()?.let{ | ||||||
|  |             "https://www.youtube.com/watch?v=$it" | ||||||
|  |         } ?: throw ErrorLoadingException("No link found") | ||||||
|  | 
 | ||||||
|  |         return newMovieLoadResponse(title, url, TvType.Movie, link) { | ||||||
|  |             this.posterUrl = document.select("div.player-content > img").attr("src") | ||||||
|  |             this.year = document.selectFirst("div.meta-bar.meta-single")?.ownText()?.filter { it.isDigit() }?.toIntOrNull() | ||||||
|  |             this.plot = document.select("div[itemprop=reviewBody] > p").text().trim() | ||||||
|  |             this.tags = document.select("div.meta-bar.meta-single > a").map { it.text() } | ||||||
|  |             this.rating = document.selectFirst("div.module div.star")?.text()?.toRatingInt() | ||||||
|  |             this.recommendations = document.select("ul.side-wrap.clear li").mapNotNull { | ||||||
|  |                 val recName = it.selectFirst("a")?.attr("title") ?: return@mapNotNull null | ||||||
|  |                 val recHref = it.selectFirst("a")!!.attr("href") | ||||||
|  |                 newMovieSearchResponse(recName, recHref, TvType.Documentary) { | ||||||
|  |                     this.posterUrl = it.selectFirst("a img")?.attr("data-src").toString() | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             addTrailer(link) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun loadLinks( | ||||||
|  |         data: String, | ||||||
|  |         isCasting: Boolean, | ||||||
|  |         subtitleCallback: (SubtitleFile) -> Unit, | ||||||
|  |         callback: (ExtractorLink) -> Unit | ||||||
|  |     ): Boolean { | ||||||
|  | 
 | ||||||
|  |         loadExtractor(data, data, subtitleCallback, callback) | ||||||
|  | 
 | ||||||
|  |         return true | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -0,0 +1,14 @@ | ||||||
|  | 
 | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.lagradost.cloudstream3.plugins.CloudstreamPlugin | ||||||
|  | import com.lagradost.cloudstream3.plugins.Plugin | ||||||
|  | import android.content.Context | ||||||
|  | 
 | ||||||
|  | @CloudstreamPlugin | ||||||
|  | class TopdocumentaryfilmsPlugin: Plugin() { | ||||||
|  |     override fun load(context: Context) { | ||||||
|  |         // All providers should be added in this manner. Please don't edit the providers list directly. | ||||||
|  |         registerMainAPI(Topdocumentaryfilms()) | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										22
									
								
								Tvtwofourseven/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								Tvtwofourseven/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,22 @@ | ||||||
|  | // use an integer for version numbers | ||||||
|  | version = 2 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | cloudstream { | ||||||
|  |     language = "en" | ||||||
|  |     // All of these properties are optional, you can safely remove them | ||||||
|  | 
 | ||||||
|  |     // description = "Lorem Ipsum" | ||||||
|  |      authors = listOf("Hexated") | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Status int as the following: | ||||||
|  |      * 0: Down | ||||||
|  |      * 1: Ok | ||||||
|  |      * 2: Slow | ||||||
|  |      * 3: Beta only | ||||||
|  |      * */ | ||||||
|  |     status = 1 // will be 3 if unspecified | ||||||
|  |     tvTypes = listOf("Live") | ||||||
|  |     iconUrl = "https://www.google.com/s2/favicons?domain=tv247.us&sz=%size%" | ||||||
|  | } | ||||||
							
								
								
									
										2
									
								
								Tvtwofourseven/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								Tvtwofourseven/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,2 @@ | ||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <manifest package="com.hexated"/> | ||||||
							
								
								
									
										114
									
								
								Tvtwofourseven/src/main/kotlin/com/hexated/Tvtwofourseven.kt
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										114
									
								
								Tvtwofourseven/src/main/kotlin/com/hexated/Tvtwofourseven.kt
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,114 @@ | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.lagradost.cloudstream3.* | ||||||
|  | import com.lagradost.cloudstream3.utils.ExtractorLink | ||||||
|  | import com.lagradost.cloudstream3.utils.M3u8Helper | ||||||
|  | import com.lagradost.cloudstream3.utils.Qualities | ||||||
|  | import org.jsoup.nodes.Element | ||||||
|  | import java.net.URI | ||||||
|  | 
 | ||||||
|  | class Tvtwofourseven : MainAPI() { | ||||||
|  |     override var mainUrl = "http://tv247.us" | ||||||
|  |     override var name = "Tv247" | ||||||
|  |     override val hasDownloadSupport = false | ||||||
|  |     override val hasMainPage = true | ||||||
|  |     override val supportedTypes = setOf( | ||||||
|  |         TvType.Live | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     override suspend fun getMainPage(page: Int, request: MainPageRequest): HomePageResponse { | ||||||
|  |         val home = listOf( | ||||||
|  |             Pair("$mainUrl/top-channels", "Top Channels"), | ||||||
|  |             Pair("$mainUrl/all-channels", "All Channels") | ||||||
|  |         ).apmap { (url,name) -> | ||||||
|  |             val home = | ||||||
|  |                 app.get(url).document.select("div.grid-items div.item").mapNotNull { item -> | ||||||
|  |                     item.toSearchResult() | ||||||
|  |                 } | ||||||
|  |             HomePageList(name, home, true) | ||||||
|  |         }.filter { it.list.isNotEmpty() } | ||||||
|  |         return HomePageResponse(home) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun Element.toSearchResult(): LiveSearchResponse? { | ||||||
|  |         return LiveSearchResponse( | ||||||
|  |             this.selectFirst("div.layer-content a")?.text() ?: return null, | ||||||
|  |             fixUrlNull(this.selectFirst("a")?.attr("href")) ?: return null, | ||||||
|  |             this@Tvtwofourseven.name, | ||||||
|  |             TvType.Live, | ||||||
|  |             fixUrlNull(this.select("img").attr("src")), | ||||||
|  |         ) | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun search(query: String): List<SearchResponse> { | ||||||
|  |         return app.post( | ||||||
|  |             "$mainUrl/wp-admin/admin-ajax.php", data = mapOf( | ||||||
|  |                 "action" to "ajaxsearchlite_search", | ||||||
|  |                 "aslp" to query, | ||||||
|  |                 "asid" to "1", | ||||||
|  |                 "options" to "qtranslate_lang=0&set_intitle=None&set_incontent=None&set_inposts=None" | ||||||
|  |             ), | ||||||
|  |             headers = mapOf("X-Requested-With" to "XMLHttpRequest") | ||||||
|  |         ).document.select("div.item").mapNotNull { | ||||||
|  |             LiveSearchResponse( | ||||||
|  |                 it.selectFirst("a")?.text() ?: return@mapNotNull null, | ||||||
|  |                 fixUrl(it.selectFirst("a")!!.attr("href")), | ||||||
|  |                 this@Tvtwofourseven.name, | ||||||
|  |                 TvType.Live, | ||||||
|  |                 fixUrlNull( | ||||||
|  |                     it.select("div.asl_image").attr("style").substringAfter("url(\"") | ||||||
|  |                         .substringBefore("\");") | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun load(url: String): LoadResponse? { | ||||||
|  |         val document = app.get(url).document | ||||||
|  |         val data = | ||||||
|  |             document.select("script").find { it.data().contains("var channelName =") }?.data() | ||||||
|  |         val baseUrl = data?.substringAfter("baseUrl = \"")?.substringBefore("\";") | ||||||
|  |         val channel = data?.substringAfter("var channelName = \"")?.substringBefore("\";") | ||||||
|  |         return LiveStreamLoadResponse( | ||||||
|  |             document.selectFirst("title")?.text()?.split("-")?.first()?.trim() ?: return null, | ||||||
|  |             url, | ||||||
|  |             this.name, | ||||||
|  |             "$baseUrl$channel.m3u8", | ||||||
|  |             fixUrlNull(document.selectFirst("img.aligncenter.jetpack-lazy-image")?.attr("src")), | ||||||
|  |             plot = document.select("address").text() | ||||||
|  |         ) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override suspend fun loadLinks( | ||||||
|  |         data: String, | ||||||
|  |         isCasting: Boolean, | ||||||
|  |         subtitleCallback: (SubtitleFile) -> Unit, | ||||||
|  |         callback: (ExtractorLink) -> Unit | ||||||
|  |     ): Boolean { | ||||||
|  | 
 | ||||||
|  |         if (URI(data).host == "cdn.espnfree.xyz") { | ||||||
|  |             M3u8Helper.generateM3u8( | ||||||
|  |                 this.name, | ||||||
|  |                 data, | ||||||
|  |                 "$mainUrl/", | ||||||
|  |                 headers = mapOf("Origin" to mainUrl, "X-Cache" to "HIT"), | ||||||
|  |             ).forEach(callback) | ||||||
|  |         } else { | ||||||
|  |             callback.invoke( | ||||||
|  |                 ExtractorLink( | ||||||
|  |                     source = name, | ||||||
|  |                     name = name, | ||||||
|  |                     url = data, | ||||||
|  |                     referer = "$mainUrl/", | ||||||
|  |                     quality = Qualities.Unknown.value, | ||||||
|  |                     isM3u8 = true, | ||||||
|  |                     headers = mapOf("Origin" to mainUrl) | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return true | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -0,0 +1,14 @@ | ||||||
|  | 
 | ||||||
|  | package com.hexated | ||||||
|  | 
 | ||||||
|  | import com.lagradost.cloudstream3.plugins.CloudstreamPlugin | ||||||
|  | import com.lagradost.cloudstream3.plugins.Plugin | ||||||
|  | import android.content.Context | ||||||
|  | 
 | ||||||
|  | @CloudstreamPlugin | ||||||
|  | class TvtwofoursevenPlugin: Plugin() { | ||||||
|  |     override fun load(context: Context) { | ||||||
|  |         // All providers should be added in this manner. Please don't edit the providers list directly. | ||||||
|  |         registerMainAPI(Tvtwofourseven()) | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										27
									
								
								UakinoProvider/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								UakinoProvider/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,27 @@ | ||||||
|  | // use an integer for version numbers | ||||||
|  | version = 1 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | cloudstream { | ||||||
|  |     language = "uk" | ||||||
|  |     // All of these properties are optional, you can safely remove them | ||||||
|  | 
 | ||||||
|  |     // description = "Lorem Ipsum" | ||||||
|  |     authors = listOf("Hexated") | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Status int as the following: | ||||||
|  |      * 0: Down | ||||||
|  |      * 1: Ok | ||||||
|  |      * 2: Slow | ||||||
|  |      * 3: Beta only | ||||||
|  |      * */ | ||||||
|  |     status = 1 // will be 3 if unspecified | ||||||
|  |     tvTypes = listOf( | ||||||
|  |         "Anime", | ||||||
|  |         "TvSeries", | ||||||
|  |         "Movie", | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     iconUrl = "https://www.google.com/s2/favicons?domain=uakino.club&sz=%size%" | ||||||
|  | } | ||||||
							
								
								
									
										2
									
								
								UakinoProvider/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								UakinoProvider/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,2 @@ | ||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <manifest package="com.hexated"/> | ||||||
Some files were not shown because too many files have changed in this diff Show more
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue