mirror of
				https://github.com/Jacekun/cs3xxx-repo.git
				synced 2024-08-14 23:57:09 +00:00 
			
		
		
		
	Fucked up
This commit is contained in:
		
							parent
							
								
									8f8033178d
								
							
						
					
					
						commit
						e9f33a1f8f
					
				
					 42 changed files with 335 additions and 1548 deletions
				
			
		|  | @ -1,12 +1,12 @@ | ||||||
| // use an integer for version numbers | // use an integer for version numbers | ||||||
| version = 6 | version = 1 | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| cloudstream { | cloudstream { | ||||||
|     // All of these properties are optional, you can safely remove them |     // All of these properties are optional, you can safely remove them | ||||||
| 
 | 
 | ||||||
|     description = "sxyprn" |     description = "" | ||||||
|     authors = listOf("Coxju") |     authors = listOf("Jace") | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|     * Status int as the following: |     * Status int as the following: | ||||||
|  | @ -15,14 +15,12 @@ cloudstream { | ||||||
|     * 2: Slow |     * 2: Slow | ||||||
|     * 3: Beta only |     * 3: Beta only | ||||||
|     * */ |     * */ | ||||||
|     status = 1 // will be 3 if unspecified |     status = 0 // will be 3 if unspecified | ||||||
| 
 | 
 | ||||||
|     // List of video source types. Users are able to filter for extensions in a given category. |     // 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: |     // You can find a list of avaliable types here: | ||||||
|     // https://recloudstream.github.io/cloudstream/html/app/com.lagradost.cloudstream3/-tv-type/index.html |     // https://recloudstream.github.io/cloudstream/html/app/com.lagradost.cloudstream3/-tv-type/index.html | ||||||
|     tvTypes = listOf("NSFW") |     tvTypes = listOf("NSFW") | ||||||
| 
 | 
 | ||||||
|     iconUrl = "https://www.google.com/s2/favicons?domain=sxyprn.com&sz=%size%" |     iconUrl = "https://www.google.com/s2/favicons?domain=example.com&sz=%size%" | ||||||
| 
 |  | ||||||
|     language = "en" |  | ||||||
| } | } | ||||||
|  | @ -1,2 +1,2 @@ | ||||||
| <?xml version="1.0" encoding="utf-8"?> | <?xml version="1.0" encoding="utf-8"?> | ||||||
| <manifest package="com.coxjud"/> | <manifest package="com.lagradost"/> | ||||||
							
								
								
									
										9
									
								
								Example/src/main/kotlin/com/jacekun/Example.kt
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								Example/src/main/kotlin/com/jacekun/Example.kt
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,9 @@ | ||||||
|  | package com.jacekun | ||||||
|  | 
 | ||||||
|  | import com.lagradost.cloudstream3.MainAPI | ||||||
|  | import com.lagradost.cloudstream3.TvType | ||||||
|  | 
 | ||||||
|  | class Example : MainAPI() { | ||||||
|  |     private val DEV = "DevDebug" | ||||||
|  |     private val globaltvType = TvType.Movie | ||||||
|  | } | ||||||
|  | @ -1,13 +1,13 @@ | ||||||
| package com.KillerDogeEmpire | package com.jacekun | ||||||
| 
 | 
 | ||||||
| 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 XhamsterProvider: Plugin() { | class ExamplePlugin: 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(Xhamster()) |         registerMainAPI(Example()) | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | @ -1,18 +0,0 @@ | ||||||
| version = 4 |  | ||||||
| 
 |  | ||||||
| cloudstream { |  | ||||||
|     authors     = listOf("KillerDogeEmpire, Coxju") |  | ||||||
|     language    = "en" |  | ||||||
|     description = "FullPorner is the best free full length porn video site. Choose from millions of hardcore videos that stream quickly and in high quality and only full length" |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Status int as the following: |  | ||||||
|      * 0: Down |  | ||||||
|      * 1: Ok |  | ||||||
|      * 2: Slow |  | ||||||
|      * 3: Beta only |  | ||||||
|     **/ |  | ||||||
|     status  = 1 // will be 3 if unspecified |  | ||||||
|     tvTypes = listOf("NSFW") |  | ||||||
|     iconUrl = "https://www.google.com/s2/favicons?domain=fullporner.com&sz=%size%" |  | ||||||
| } |  | ||||||
|  | @ -1,162 +0,0 @@ | ||||||
| package com.KillerDogeEmpire |  | ||||||
| 
 |  | ||||||
| import org.jsoup.nodes.Element |  | ||||||
| import com.lagradost.cloudstream3.* |  | ||||||
| import com.lagradost.cloudstream3.utils.* |  | ||||||
| import com.lagradost.cloudstream3.network.WebViewResolver |  | ||||||
| import com.lagradost.cloudstream3.LoadResponse.Companion.addActors |  | ||||||
| import org.jsoup.Jsoup |  | ||||||
| 
 |  | ||||||
| class FullPorner : MainAPI() { |  | ||||||
|     override var mainUrl              = "https://fullporner.com" |  | ||||||
|     override var name                 = "FullPorner" |  | ||||||
|     override val hasMainPage          = true |  | ||||||
|     override var lang                 = "en" |  | ||||||
|     override val hasQuickSearch       = false |  | ||||||
|     override val hasDownloadSupport   = true |  | ||||||
|     override val hasChromecastSupport = true |  | ||||||
|     override val supportedTypes       = setOf(TvType.NSFW) |  | ||||||
|     override val vpnStatus            = VPNStatus.MightBeNeeded |  | ||||||
| 
 |  | ||||||
|     override val mainPage = mainPageOf( |  | ||||||
|         "${mainUrl}/home/"                to "Featured", |  | ||||||
|         "${mainUrl}/category/amateur/"    to "Amateur", |  | ||||||
|         "${mainUrl}/category/teen/"       to "Teen", |  | ||||||
|         "${mainUrl}/category/cumshot/"    to "CumShot", |  | ||||||
|         "${mainUrl}/category/deepthroat/" to "DeepThroat", |  | ||||||
|         "${mainUrl}/category/orgasm/"     to "Orgasm", |  | ||||||
|         "${mainUrl}/category/threesome/"  to "ThreeSome", |  | ||||||
|         "${mainUrl}/category/group-sex/"  to "Group Sex", |  | ||||||
|     ) |  | ||||||
| 
 |  | ||||||
|     override suspend fun getMainPage(page: Int, request: MainPageRequest): HomePageResponse { |  | ||||||
|         val document = app.get("${request.data}${page}").document |  | ||||||
|         val home     = document.select("div.video-block div.video-card").mapNotNull { it.toSearchResult() } |  | ||||||
| 
 |  | ||||||
|         return newHomePageResponse( |  | ||||||
|             list    = HomePageList( |  | ||||||
|                 name               = request.name, |  | ||||||
|                 list               = home, |  | ||||||
|                 isHorizontalImages = true |  | ||||||
|             ), |  | ||||||
|             hasNext = true |  | ||||||
|         ) |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private fun Element.toSearchResult(): SearchResponse? { |  | ||||||
|         val title = this.selectFirst("div.video-card div.video-card-body div.video-title a")?.text() ?: return null |  | ||||||
|         val href = fixUrl(this.selectFirst("div.video-card div.video-card-body div.video-title a")!!.attr("href")) |  | ||||||
|         val posterUrl = fixUrlNull(this.select("div.video-card div.video-card-image a img").attr("data-src")) |  | ||||||
| 
 |  | ||||||
|         return newMovieSearchResponse(title, href, TvType.Movie) { this.posterUrl = posterUrl } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     override suspend fun search(query: String): List<SearchResponse> { |  | ||||||
|         val searchResponse = mutableListOf<SearchResponse>() |  | ||||||
| 
 |  | ||||||
|         for (i in 1..15) { |  | ||||||
|             val document = app.get("${mainUrl}/search?q=${query.replace(" ", "+")}&p=$i").document |  | ||||||
| 
 |  | ||||||
|             val results = document.select("div.video-block div.video-card").mapNotNull { it.toSearchResult() } |  | ||||||
| 
 |  | ||||||
|             searchResponse.addAll(results) |  | ||||||
| 
 |  | ||||||
|             if (results.isEmpty()) break |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         return searchResponse |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     override suspend fun load(url: String): LoadResponse { |  | ||||||
|         val document = app.get(url).document |  | ||||||
| 
 |  | ||||||
|         val title     = document.selectFirst("div.video-block div.single-video-left div.single-video-title h2")?.text()?.trim().toString() |  | ||||||
|         val iframeUrl = fixUrlNull(document.selectFirst("div.video-block div.single-video-left div.single-video iframe")?.attr("src")) ?: "" |  | ||||||
| 
 |  | ||||||
|         val poster: String? |  | ||||||
|         val posterHeaders: Map<String, String> |  | ||||||
|         if (iframeUrl.contains("videoh")) { |  | ||||||
|             val iframeDocument = app.get(iframeUrl, interceptor = WebViewResolver(Regex("""mydaddy"""))).document |  | ||||||
| 
 |  | ||||||
|             val videoHtml = iframeDocument.selectXpath("//script[contains(text(),'poster')]").first()?.html()?.substringAfter("else{ \$(\"#jw\").html(\"")?.substringBefore("\");}if(hasAdblock)")?.replace("\\", "") |  | ||||||
|             val video     = Jsoup.parse(videoHtml.toString()).selectFirst("video") |  | ||||||
| 
 |  | ||||||
|             poster        = fixUrlNull(video?.attr("poster")) |  | ||||||
|             posterHeaders = mapOf(Pair("referer", "https://mydaddy.cc/")) |  | ||||||
|         } else { |  | ||||||
|             val iframeDocument = app.get(iframeUrl).document |  | ||||||
|             val videoDocument  = Jsoup.parse("<video" + iframeDocument.selectXpath("//script[contains(text(),'\$(\"#jw\").html(')]")[0]?.toString()?.replace("\\", "")?.substringAfter("<video")?.substringBefore("</video>") + "</video>") |  | ||||||
| 
 |  | ||||||
|             poster        = fixUrlNull(videoDocument.selectFirst("video")?.attr("poster").toString()) |  | ||||||
|             posterHeaders = mapOf(Pair("referer", "https://xiaoshenke.net/")) |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         val tags            = document.select("div.video-blockdiv.single-video-left div.single-video-title p.tag-link span a").map { it.text() } |  | ||||||
|         val description     = document.selectFirst("div.video-block div.single-video-left div.single-video-title h2")?.text()?.trim().toString() |  | ||||||
|         val actors          = document.select("div.video-block div.single-video-left div.single-video-info-content p a").map { it.text() } |  | ||||||
|         val recommendations = document.select("div.video-block div.video-recommendation div.video-card").mapNotNull { it.toSearchResult() } |  | ||||||
| 
 |  | ||||||
|         return newMovieLoadResponse(title, url, TvType.NSFW, url) { |  | ||||||
|             this.posterUrl       = poster |  | ||||||
|             this.posterHeaders   = posterHeaders |  | ||||||
|             this.plot            = description |  | ||||||
|             this.tags            = tags |  | ||||||
|             this.recommendations = recommendations |  | ||||||
|             addActors(actors) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     override suspend fun loadLinks(data: String, isCasting: Boolean, subtitleCallback: (SubtitleFile) -> Unit, callback: (ExtractorLink) -> Unit): Boolean { |  | ||||||
|         val document    = app.get(data).document |  | ||||||
| 
 |  | ||||||
|         val iframeUrl   = fixUrlNull(document.selectFirst("div.video-block div.single-video-left div.single-video iframe")?.attr("src")) ?: "" |  | ||||||
| 
 |  | ||||||
|         val extlinkList = mutableListOf<ExtractorLink>() |  | ||||||
|         if (iframeUrl.contains("videoh")) { |  | ||||||
|             val iframeDocument = app.get(iframeUrl, interceptor = WebViewResolver(Regex("""mydaddy"""))).document |  | ||||||
|             val videoDocument  = Jsoup.parse("<video" + iframeDocument.selectXpath("//script[contains(text(),'\$(\"#jw\").html(')]").first()?.toString()?.replace("\\", "")?.substringAfter("<video")?.substringAfter("<video")?.substringBefore("</video>") + "</video>") |  | ||||||
| 
 |  | ||||||
|             videoDocument.select("source").map { res ->  |  | ||||||
|                 extlinkList.add(ExtractorLink( |  | ||||||
|                     name, |  | ||||||
|                     name, |  | ||||||
|                     fixUrl(res.attr("src")), |  | ||||||
|                     referer = data, |  | ||||||
|                     quality = Regex("(\\d+.)").find(res.attr("title"))?.groupValues?.get(1).let { getQualityFromName(it) } |  | ||||||
|                 ))  |  | ||||||
|             } |  | ||||||
|         } else if (iframeUrl.contains("xiaoshenke")) { |  | ||||||
|             val iframeDocument = app.get(iframeUrl).document |  | ||||||
|             val videoID        = Regex("""var id = \"(.+?)\"""").find(iframeDocument.html())?.groupValues?.get(1) |  | ||||||
| 
 |  | ||||||
|             val pornTrexDocument = app.get("https://www.porntrex.com/embed/${videoID}").document |  | ||||||
|             val video_url = fixUrlNull(Regex("""video_url: \'(.+?)\',""").find(pornTrexDocument.html())?.groupValues?.get(1)) |  | ||||||
|             if (video_url != null) { |  | ||||||
|                 extlinkList.add(ExtractorLink( |  | ||||||
|                     name, |  | ||||||
|                     name, |  | ||||||
|                     video_url, |  | ||||||
|                     referer = data, |  | ||||||
|                     quality = Qualities.Unknown.value |  | ||||||
|                 )) |  | ||||||
|             } |  | ||||||
|         } else { |  | ||||||
|             val iframeDocument = app.get(iframeUrl).document |  | ||||||
|             val videoDocument  = Jsoup.parse("<video" + iframeDocument.selectXpath("//script[contains(text(),'\$(\"#jw\").html(')]").first()?.toString()?.replace("\\", "")?.substringAfter("<video")?.substringBefore("</video>") + "</video>") |  | ||||||
| 
 |  | ||||||
|             videoDocument.select("source").map { res ->  |  | ||||||
|                 extlinkList.add(ExtractorLink( |  | ||||||
|                     this.name, |  | ||||||
|                     this.name, |  | ||||||
|                     fixUrl(res.attr("src")), |  | ||||||
|                     referer = mainUrl, |  | ||||||
|                     quality = Regex("(\\d+.)").find(res.attr("title"))?.groupValues?.get(1).let { getQualityFromName(it) } |  | ||||||
|                 ))  |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         extlinkList.forEach(callback) |  | ||||||
| 
 |  | ||||||
|         return true |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -1,12 +0,0 @@ | ||||||
| package com.KillerDogeEmpire |  | ||||||
| 
 |  | ||||||
| import android.content.Context |  | ||||||
| import com.lagradost.cloudstream3.plugins.CloudstreamPlugin |  | ||||||
| import com.lagradost.cloudstream3.plugins.Plugin |  | ||||||
| 
 |  | ||||||
| @CloudstreamPlugin |  | ||||||
| class FullPornerProvider : Plugin() { |  | ||||||
|     override fun load(context: Context) { |  | ||||||
|         registerMainAPI(FullPorner()) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -1,28 +0,0 @@ | ||||||
| // use an integer for version numbers |  | ||||||
| version = 5 |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| cloudstream { |  | ||||||
|     // All of these properties are optional, you can safely remove them |  | ||||||
| 
 |  | ||||||
|     description = "GoodPorn" |  | ||||||
|     authors = listOf(" KillerDogeEmpire, Stormunblessed, Jace, Hexated, Coxju") |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|     * 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("NSFW") |  | ||||||
| 
 |  | ||||||
|     iconUrl = "https://www.google.com/s2/favicons?domain=goodporn.to&sz=%size%" |  | ||||||
| 
 |  | ||||||
|     language = "en" |  | ||||||
| } |  | ||||||
|  | @ -1,2 +0,0 @@ | ||||||
| <?xml version="1.0" encoding="utf-8"?> |  | ||||||
| <manifest package="com.KillerDogeEmpire"/> |  | ||||||
|  | @ -1,127 +0,0 @@ | ||||||
| package com.KillerDogeEmpire |  | ||||||
| 
 |  | ||||||
| import com.lagradost.cloudstream3.* |  | ||||||
| import com.lagradost.cloudstream3.LoadResponse.Companion.addActors |  | ||||||
| import com.lagradost.cloudstream3.utils.ExtractorLink |  | ||||||
| import com.lagradost.cloudstream3.utils.getQualityFromName |  | ||||||
| import org.jsoup.nodes.Element |  | ||||||
| import java.util.* |  | ||||||
| 
 |  | ||||||
| class GoodPorn : MainAPI() { |  | ||||||
|     override var mainUrl = "https://goodporn.to" |  | ||||||
|     override var name = "GoodPorn" |  | ||||||
|     override val hasMainPage = true |  | ||||||
|     override val hasDownloadSupport = true |  | ||||||
|     override val vpnStatus = VPNStatus.MightBeNeeded |  | ||||||
|     override val supportedTypes = setOf(TvType.NSFW) |  | ||||||
| 
 |  | ||||||
|     override val mainPage = mainPageOf( |  | ||||||
|         "$mainUrl/?mode=async&function=get_block&block_id=list_videos_most_recent_videos&sort_by=post_date&from=" to "New Videos", |  | ||||||
|         "$mainUrl/?mode=async&function=get_block&block_id=list_videos_most_recent_videos&sort_by=video_viewed&from=" to "Most Viewed Videos", |  | ||||||
|         "$mainUrl/?mode=async&function=get_block&block_id=list_videos_most_recent_videos&sort_by=rating&from=" to "Top Rated Videos ", |  | ||||||
|         "$mainUrl/?mode=async&function=get_block&block_id=list_videos_most_recent_videos&sort_by=most_commented&from=" to "Most Commented Videos", |  | ||||||
|         "$mainUrl/?mode=async&function=get_block&block_id=list_videos_most_recent_videos&sort_by=duration&from=" to "Longest Videos", |  | ||||||
|         "$mainUrl/sites/fitness-rooms/?mode=async&function=get_block&block_id=list_videos_common_videos_list&sort_by=post_date&from=" to "Fitness Rooms", |  | ||||||
|         "$mainUrl/sites/public-agent/?mode=async&function=get_block&block_id=list_videos_common_videos_list&sort_by=post_date&from=" to "Public Agent", |  | ||||||
|         "$mainUrl/sites/massage-rooms/?mode=async&function=get_block&block_id=list_videos_common_videos_list&sort_by=post_date&from=" to "Massage Rooms", |  | ||||||
|         "$mainUrl/sites/dane-jones/?mode=async&function=get_block&block_id=list_videos_common_videos_list&sort_by=post_date&from=" to "Dane Jones", |  | ||||||
|         "$mainUrl/channels/brazzers/?mode=async&function=get_block&block_id=list_videos_common_videos_list&sort_by=post_date&from=" to "Brazzers", |  | ||||||
|         "$mainUrl/channels/digitalplayground/?mode=async&function=get_block&block_id=list_videos_common_videos_list&sort_by=post_date&from=" to "Digital Playground", |  | ||||||
|         "$mainUrl/channels/realitykings/?mode=async&function=get_block&block_id=list_videos_common_videos_list&sort_by=post_date&from=" to "Realitykings", |  | ||||||
|         "$mainUrl/channels/babes-network/?mode=async&function=get_block&block_id=list_videos_common_videos_list&sort_by=post_date&from=" to "Babes Network", |  | ||||||
|     ) |  | ||||||
| 
 |  | ||||||
|     override suspend fun getMainPage( |  | ||||||
|         page: Int, request: MainPageRequest |  | ||||||
|     ): HomePageResponse { |  | ||||||
|         val document = app.get(request.data + page).document |  | ||||||
|         val home = |  | ||||||
|             document.select("div#list_videos_most_recent_videos_items div.item, div#list_videos_common_videos_list_items div.item") |  | ||||||
|                 .mapNotNull { |  | ||||||
|                     it.toSearchResult() |  | ||||||
|                 } |  | ||||||
|         return newHomePageResponse( |  | ||||||
|             list = HomePageList( |  | ||||||
|                 name = request.name, list = home, isHorizontalImages = true |  | ||||||
|             ), hasNext = true |  | ||||||
|         ) |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private fun Element.toSearchResult(): SearchResponse? { |  | ||||||
|         val title = this.selectFirst("strong.title")?.text() ?: return null |  | ||||||
|         val href = fixUrl(this.selectFirst("a")!!.attr("href")) |  | ||||||
|         val posterUrl = fixUrlNull(this.select("div.img > img").attr("data-original")) |  | ||||||
|         return newMovieSearchResponse(title, href, TvType.Movie) { |  | ||||||
|             this.posterUrl = posterUrl |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     override suspend fun search(query: String): List<SearchResponse> { |  | ||||||
|         val searchResponse = mutableListOf<SearchResponse>() |  | ||||||
|         for (i in 1..15) { |  | ||||||
|             val document = app.get( |  | ||||||
|                 "$mainUrl/search/nikki-benz/?mode=async&function=get_block&block_id=list_videos_videos_list_search_result&q=$query&category_ids=&sort_by=&from_videos=$i&from_albums=$i", |  | ||||||
|                 headers = mapOf("X-Requested-With" to "XMLHttpRequest") |  | ||||||
|             ).document |  | ||||||
|             val results = |  | ||||||
|                 document.select("div#list_videos_videos_list_search_result_items div.item") |  | ||||||
|                     .mapNotNull { |  | ||||||
|                         it.toSearchResult() |  | ||||||
|                     } |  | ||||||
|             searchResponse.addAll(results) |  | ||||||
|             if (results.isEmpty()) break |  | ||||||
|         } |  | ||||||
|         return searchResponse |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     override suspend fun load(url: String): LoadResponse { |  | ||||||
|         val document = app.get(url).document |  | ||||||
| 
 |  | ||||||
|         val title = document.selectFirst("div.headline > h1")?.text()?.trim().toString() |  | ||||||
|         val poster = |  | ||||||
|             fixUrlNull(document.selectFirst("meta[property=og:image]")?.attr("content").toString()) |  | ||||||
|         val tags = document.select("div.info div:nth-child(5) > a").map { it.text() } |  | ||||||
|         val description = document.select("div.info div:nth-child(2)").text().trim() |  | ||||||
|         val actors = document.select("div.info div:nth-child(6) > a").map { it.text() } |  | ||||||
|         val recommendations = |  | ||||||
|             document.select("div#list_videos_related_videos_items div.item").mapNotNull { |  | ||||||
|                 it.toSearchResult() |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|         return newMovieLoadResponse(title, url, TvType.NSFW, url) { |  | ||||||
|             this.posterUrl = poster |  | ||||||
|             this.plot = description |  | ||||||
|             this.tags = tags |  | ||||||
|             addActors(actors) |  | ||||||
|             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 extlinkList = mutableListOf<ExtractorLink>() |  | ||||||
|         document.select("div.info div:last-child a").map { res -> |  | ||||||
|             extlinkList.add( |  | ||||||
|                 ExtractorLink( |  | ||||||
|                     this.name, |  | ||||||
|                     this.name, |  | ||||||
|                     res.attr("href") |  | ||||||
|                         .replace(Regex("\\?download\\S+.mp4&"), "?") + "&rnd=${Date().time}", |  | ||||||
|                     referer = data, |  | ||||||
|                     quality = Regex("(\\d+.),").find(res.text())?.groupValues?.get(1) |  | ||||||
|                         .let { getQualityFromName(it) }, |  | ||||||
|                     headers = mapOf("Range" to "bytes=0-"), |  | ||||||
|                 ) |  | ||||||
|             ) |  | ||||||
|         } |  | ||||||
|         extlinkList.forEach(callback) |  | ||||||
|         return true |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
| } |  | ||||||
|  | @ -1,14 +0,0 @@ | ||||||
| package com.KillerDogeEmpire |  | ||||||
| 
 |  | ||||||
| import android.content.Context |  | ||||||
| import com.KillerDogeEmpire.GoodPorn |  | ||||||
| import com.lagradost.cloudstream3.plugins.CloudstreamPlugin |  | ||||||
| import com.lagradost.cloudstream3.plugins.Plugin |  | ||||||
| 
 |  | ||||||
| @CloudstreamPlugin |  | ||||||
| class GoodPornProvider : Plugin() { |  | ||||||
|     override fun load(context: Context) { |  | ||||||
|         // All providers should be added in this manner. Please don't edit the providers list directly. |  | ||||||
|         registerMainAPI(GoodPorn()) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -1,12 +1,12 @@ | ||||||
| // use an integer for version numbers | // use an integer for version numbers | ||||||
| version = 6 | version = 5 | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| cloudstream { | cloudstream { | ||||||
|     // All of these properties are optional, you can safely remove them |     // All of these properties are optional, you can safely remove them | ||||||
| 
 | 
 | ||||||
|     description = "" |     description = "" | ||||||
|     authors = listOf("KillerDogeEmpire, Jace") |     authors = listOf("Jace") | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|     * Status int as the following: |     * Status int as the following: | ||||||
|  |  | ||||||
|  | @ -1,99 +1,101 @@ | ||||||
| package com.jacekun | package com.jacekun | ||||||
| 
 | 
 | ||||||
| 
 | import android.util.Log | ||||||
| import com.fasterxml.jackson.annotation.JsonProperty | import com.fasterxml.jackson.annotation.JsonProperty | ||||||
| import com.lagradost.cloudstream3.* | import com.lagradost.cloudstream3.* | ||||||
| import com.lagradost.cloudstream3.LoadResponse.Companion.addTrailer | import com.lagradost.cloudstream3.mvvm.logError | ||||||
| import com.lagradost.cloudstream3.utils.* | import com.lagradost.cloudstream3.utils.ExtractorLink | ||||||
| import okhttp3.FormBody | import com.lagradost.cloudstream3.utils.getQualityFromName | ||||||
| import org.jsoup.nodes.Element | import org.jsoup.select.Elements | ||||||
| 
 |  | ||||||
| 
 | 
 | ||||||
| class HentaiHaven : MainAPI() { | class HentaiHaven : MainAPI() { | ||||||
|     override var mainUrl = "https://hentaihaven.xxx" |     private val globalTvType = TvType.NSFW | ||||||
|     override var name = "Hentai Haven" |     override var name = "Hentai Haven" | ||||||
|     override val hasMainPage = true |     override var mainUrl = "https://hentaihaven.xxx" | ||||||
|     override var lang = "en" |     override val supportedTypes = setOf(TvType.NSFW) | ||||||
|     override val hasDownloadSupport = true |     override val hasDownloadSupport = false | ||||||
| 
 |     override val hasMainPage= true | ||||||
|     override val supportedTypes = setOf( |     override val hasQuickSearch = false | ||||||
|         TvType.NSFW) |  | ||||||
| 
 |  | ||||||
|     override val mainPage = mainPageOf( |  | ||||||
|         "?m_orderby=new-manga" to "New", |  | ||||||
|         "?m_orderby=views" to "Most Views", |  | ||||||
|         "?m_orderby=rating" to "Rating", |  | ||||||
|         "?m_orderby=alphabet" to "A-Z", |  | ||||||
|     ) |  | ||||||
| 
 | 
 | ||||||
|     override suspend fun getMainPage( |     override suspend fun getMainPage( | ||||||
|         page: Int, |         page: Int, | ||||||
|         request: MainPageRequest |         request: MainPageRequest | ||||||
|     ): HomePageResponse { |     ): HomePageResponse { | ||||||
|         val document = app.get("$mainUrl/page/$page/${request.data}").document |         val doc = app.get(mainUrl).document | ||||||
|         val home = |         val all = ArrayList<HomePageList>() | ||||||
|             document.select("div.page-listing-item div.col-6.col-md-zarat.badge-pos-1").mapNotNull { |  | ||||||
|                 it.toSearchResult() |  | ||||||
|             } |  | ||||||
|         return newHomePageResponse(request.name, home) |  | ||||||
|     } |  | ||||||
| 
 | 
 | ||||||
|     private fun Element.toSearchResult(): AnimeSearchResponse? { |         doc.getElementsByTag("body").select("div.c-tabs-item") | ||||||
|         val href = fixUrl(this.selectFirst("a")!!.attr("href")) |             .select("div.vraven_home_slider").forEach { it2 -> | ||||||
|         val title = |                 // Fetch row title | ||||||
|             this.selectFirst("h3 a, h5 a")?.text()?.trim() ?: this.selectFirst("a")?.attr("title") |                 val title = it2?.select("div.home_slider_header")?.text() ?: "Unnamed Row" | ||||||
|             ?: return null |                 // Fetch list of items and map | ||||||
|         val posterUrl = fixUrlNull(this.selectFirst("img")?.attr("src")) |                 it2.select("div.page-content-listing div.item.vraven_item.badge-pos-1").let { inner -> | ||||||
|         val episode = this.selectFirst("span.chapter.font-meta a")?.text()?.filter { it.isDigit() } |  | ||||||
|             ?.toIntOrNull() |  | ||||||
| 
 | 
 | ||||||
|         return newAnimeSearchResponse(title, href, TvType.Anime) { |                     all.add( | ||||||
|             this.posterUrl = posterUrl |                         HomePageList( | ||||||
|             addSub(episode) |                             name = title, | ||||||
|  |                             list = inner.getResults(this.name), | ||||||
|  |                             isHorizontalImages = false | ||||||
|  |                         ) | ||||||
|  |                     ) | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|  |         return HomePageResponse(all) | ||||||
|  |     } | ||||||
| 
 | 
 | ||||||
|     override suspend fun search(query: String): List<SearchResponse> { |     override suspend fun search(query: String): List<SearchResponse> { | ||||||
|         val link = "$mainUrl/?s=$query&post_type=wp-manga" |         val searchUrl = "${mainUrl}/?s=${query}&post_type=wp-manga" | ||||||
|         val document = app.get(link).document |         return app.get(searchUrl).document | ||||||
| 
 |             .select("div.c-tabs-item div.row.c-tabs-item__content") | ||||||
|         return document.select("div.c-tabs-item > div.c-tabs-item__content").mapNotNull { |             .getResults(this.name) | ||||||
|             it.toSearchResult() |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     override suspend fun load(url: String): LoadResponse? { |     override suspend fun load(url: String): LoadResponse { | ||||||
|         val document = app.get(url).document |         //TODO: Load polishing | ||||||
|  |         val doc = app.get(url).document | ||||||
|  |         //Log.i(this.name, "Result => (url) ${url}") | ||||||
|  |         val poster = doc.select("meta[property=og:image]") | ||||||
|  |             .firstOrNull()?.attr("content") | ||||||
|  |         val title = doc.select("meta[name=title]") | ||||||
|  |             .firstOrNull()?.attr("content") | ||||||
|  |             ?.toString() ?: "" | ||||||
|  |         val descript = doc.select("div.description-summary").text() | ||||||
| 
 | 
 | ||||||
|         val title = document.selectFirst("div.post-title h1")?.text()?.trim() ?: return null |         val body = doc.getElementsByTag("body") | ||||||
|         val poster = document.select("div.summary_image img").attr("src") |         val episodes = body.select("div.page-content-listing.single-page") | ||||||
|         val tags = document.select("div.genres-content > a").map { it.text() } |             .first()?.select("li") | ||||||
| 
 | 
 | ||||||
|         val description = document.select("div.description-summary p").text().trim() |         val year = episodes?.last() | ||||||
|         val trailer = document.selectFirst("a.trailerbutton")?.attr("href") |             ?.selectFirst("span.chapter-release-date") | ||||||
|  |             ?.text()?.trim()?.takeLast(4)?.toIntOrNull() | ||||||
| 
 | 
 | ||||||
|         val episodes = document.select("div.listing-chapters_wrap ul li").mapNotNull { |         val episodeList = episodes?.mapNotNull { | ||||||
|             val name = it.selectFirst("a")?.text() ?: return@mapNotNull null |             val innerA = it?.selectFirst("a") ?: return@mapNotNull null | ||||||
|             val image = fixUrlNull(it.selectFirst("a img")?.attr("src")) |             val eplink = innerA.attr("href") ?: return@mapNotNull null | ||||||
|             val link = fixUrlNull(it.selectFirst("a")?.attr("href")) ?: return@mapNotNull null |             val epCount = innerA.text().trim().filter { a -> a.isDigit() }.toIntOrNull() | ||||||
|             Episode(link, name, posterUrl = image) |             val imageEl = innerA.selectFirst("img") | ||||||
|         }.reversed() |             val epPoster = imageEl?.attr("src") ?: imageEl?.attr("data-src") | ||||||
| 
 |             Episode( | ||||||
|         val recommendations = |                 name = innerA.text(), | ||||||
|             document.select("div.row div.col-6.col-md-zarat").mapNotNull { |                 data = eplink, | ||||||
|                 it.toSearchResult() |                 posterUrl = epPoster, | ||||||
|             } |                 episode = epCount, | ||||||
| 
 |             ) | ||||||
|         return newAnimeLoadResponse(title, url, TvType.NSFW) { |         } ?: listOf() | ||||||
|             engName = title |  | ||||||
|             posterUrl = poster |  | ||||||
|             addEpisodes(DubStatus.Subbed, episodes) |  | ||||||
|             plot = description |  | ||||||
|             this.tags = tags |  | ||||||
|             this.recommendations = recommendations |  | ||||||
|             addTrailer(trailer) |  | ||||||
|         } |  | ||||||
| 
 | 
 | ||||||
|  |         //Log.i(this.name, "Result => (id) ${id}") | ||||||
|  |         return AnimeLoadResponse( | ||||||
|  |             name = title, | ||||||
|  |             url = url, | ||||||
|  |             apiName = this.name, | ||||||
|  |             type = globalTvType, | ||||||
|  |             posterUrl = poster, | ||||||
|  |             year = year, | ||||||
|  |             plot = descript, | ||||||
|  |             episodes = mutableMapOf( | ||||||
|  |                 Pair(DubStatus.Subbed, episodeList.reversed()) | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     override suspend fun loadLinks( |     override suspend fun loadLinks( | ||||||
|  | @ -103,66 +105,120 @@ class HentaiHaven : MainAPI() { | ||||||
|         callback: (ExtractorLink) -> Unit |         callback: (ExtractorLink) -> Unit | ||||||
|     ): Boolean { |     ): Boolean { | ||||||
| 
 | 
 | ||||||
|         val doc = app.get(data).document |         try { | ||||||
|         val meta = doc.selectFirst("meta[itemprop=thumbnailUrl]")?.attr("content")?.substringAfter("/hh/")?.substringBefore("/") ?: return false |             Log.i(name, "Loading iframe") | ||||||
|         doc.select("div.player_logic_item iframe").attr("src").let { iframe -> |             val requestLink = "${mainUrl}/wp-content/plugins/player-logic/api.php" | ||||||
|             val document = app.get(iframe, referer = data).text |             val action = "zarat_get_data_player_ajax" | ||||||
|             val en = Regex("var\\sen\\s=\\s'(\\S+)';").find(document)?.groupValues?.getOrNull(1) |             val reA = Regex("(?<=var en =)(.*?)(?=';)", setOf(RegexOption.DOT_MATCHES_ALL)) | ||||||
|             val iv = Regex("var\\siv\\s=\\s'(\\S+)';").find(document)?.groupValues?.getOrNull(1) |             val reB = Regex("(?<=var iv =)(.*?)(?=';)", setOf(RegexOption.DOT_MATCHES_ALL)) | ||||||
| 
 | 
 | ||||||
|             val body = FormBody.Builder() |             app.get(data).document.selectFirst("div.player_logic_item iframe") | ||||||
|                 .addEncoded("action", "zarat_get_data_player_ajax") |                 ?.attr("src")?.let { epLink -> | ||||||
|                 .addEncoded("a", "$en") |  | ||||||
|                 .addEncoded("b", "$iv") |  | ||||||
|                 .build() |  | ||||||
| 
 | 
 | ||||||
|             app.post( |                     Log.i(name, "Loading ep link => $epLink") | ||||||
|                 "$mainUrl/wp-content/plugins/player-logic/api.php", |                     val scrAppGet = app.get(epLink, referer = data) | ||||||
| //                data = mapOf( |                     val scrDoc = scrAppGet.document.getElementsByTag("script").toString() | ||||||
| //                    "action" to "zarat_get_data_player_ajax", |                     //Log.i(name, "Loading scrDoc => (${scrAppGet.code}) $scrDoc") | ||||||
| //                    "a" to "$en", |                     if (scrDoc.isNotBlank()) { | ||||||
| //                    "b" to "$iv" |                         //en | ||||||
| //                ), |                         val a = reA.find(scrDoc)?.groupValues?.getOrNull(1) | ||||||
|                 requestBody = body, |                             ?.trim()?.removePrefix("'") ?: "" | ||||||
| //                headers = mapOf("Sec-Fetch-Mode" to "cors") |                         //iv | ||||||
|             ).parsedSafe<Response>()?.data?.sources?.map { res -> |                         val b = reB.find(scrDoc)?.groupValues?.getOrNull(1) | ||||||
| //                M3u8Helper.generateM3u8( |                             ?.trim()?.removePrefix("'") ?: "" | ||||||
| //                    this.name, | 
 | ||||||
| //                    res.src ?: return@map null, |                         Log.i(name, "a => $a") | ||||||
| //                    referer = "$mainUrl/", |                         Log.i(name, "b => $b") | ||||||
| //                    headers = mapOf( | 
 | ||||||
| //                        "Origin" to mainUrl, |                         val doc = app.post( | ||||||
| //                    ) |                             url = requestLink, | ||||||
| //                ).forEach(callback) |                             headers = mapOf( | ||||||
|  | //                              Pair("mode", "cors"), | ||||||
|  | //                              Pair("Content-Type", "multipart/form-data"), | ||||||
|  | //                              Pair("Origin", mainUrl), | ||||||
|  | //                              Pair("Host", mainUrl.split("//").last()), | ||||||
|  |                                 Pair("User-Agent", USER_AGENT), | ||||||
|  |                                 Pair("Sec-Fetch-Mode", "cors") | ||||||
|  |                             ), | ||||||
|  |                             data = mapOf( | ||||||
|  |                                 Pair("action", action), | ||||||
|  |                                 Pair("a", a), | ||||||
|  |                                 Pair("b", b) | ||||||
|  |                             ) | ||||||
|  |                         ) | ||||||
|  |                         Log.i(name, "Response (${doc.code}) => ${doc.text}") | ||||||
|  |                         //AppUtils.tryParseJson<ResponseJson?>(doc.text) | ||||||
|  |                         doc.parsedSafe<ResponseJson>()?.data?.sources?.map { m3src -> | ||||||
|  |                             val m3srcFile = m3src.src ?: return@map null | ||||||
|  |                             val label = m3src.label ?: "" | ||||||
|  |                             Log.i(name, "M3u8 link: $m3srcFile") | ||||||
|                             callback.invoke( |                             callback.invoke( | ||||||
|                                 ExtractorLink( |                                 ExtractorLink( | ||||||
|                         this.name, |                                     name = "$name m3u8", | ||||||
|                         this.name, |                                     source = "$name m3u8", | ||||||
|                         res.src?.replace("/hh//", "/hh/$meta/") ?: return@map null, |                                     url = m3srcFile, | ||||||
|                         referer = "", |                                     referer = "$mainUrl/", | ||||||
|                         quality = Qualities.Unknown.value, |                                     quality = getQualityFromName(label), | ||||||
|                                     isM3u8 = true |                                     isM3u8 = true | ||||||
|                                 ) |                                 ) | ||||||
|                             ) |                             ) | ||||||
|                         } |                         } | ||||||
|                     } |                     } | ||||||
| 
 |                 } | ||||||
|  |         } catch (e: Exception) { | ||||||
|  |             Log.i(name, "Error => $e") | ||||||
|  |             logError(e) | ||||||
|  |             return false | ||||||
|  |         } | ||||||
|         return true |         return true | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     data class Response( |     private fun Elements?.getResults(apiName: String): List<AnimeSearchResponse> { | ||||||
|         @JsonProperty("data") val data: Data? = null, |         return this?.mapNotNull { | ||||||
|  |             val innerDiv = it.select("div").firstOrNull() | ||||||
|  |             val firstA = innerDiv?.selectFirst("a") | ||||||
|  |             val link = fixUrlNull(firstA?.attr("href")) ?: return@mapNotNull null | ||||||
|  |             val name = firstA?.attr("title") ?: "<No Title>" | ||||||
|  |             val year = innerDiv?.selectFirst("span.c-new-tag")?.selectFirst("a") | ||||||
|  |                 ?.attr("title")?.takeLast(4)?.toIntOrNull() | ||||||
|  | 
 | ||||||
|  |             val imageDiv = firstA?.selectFirst("img") | ||||||
|  |             var image = imageDiv?.attr("src") | ||||||
|  |             if (image.isNullOrBlank()) { | ||||||
|  |                 image = imageDiv?.attr("data-src") | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             val latestEp = innerDiv?.selectFirst("div.list-chapter") | ||||||
|  |                 ?.selectFirst("div.chapter-item") | ||||||
|  |                 ?.selectFirst("a") | ||||||
|  |                 ?.text() | ||||||
|  |                 ?.filter { a -> a.isDigit() } | ||||||
|  |                 ?.toIntOrNull() ?: 0 | ||||||
|  |             val dubStatus = mutableMapOf( | ||||||
|  |                 Pair(DubStatus.Subbed, latestEp) | ||||||
|             ) |             ) | ||||||
| 
 | 
 | ||||||
|     data class Data( |             AnimeSearchResponse( | ||||||
|         @JsonProperty("sources") val sources: ArrayList<Sources>? = arrayListOf(), |                 name = name, | ||||||
|  |                 url = link, | ||||||
|  |                 apiName = apiName, | ||||||
|  |                 type = globalTvType, | ||||||
|  |                 posterUrl = image, | ||||||
|  |                 year = year, | ||||||
|  |                 episodes = dubStatus | ||||||
|             ) |             ) | ||||||
|  |         } ?: listOf() | ||||||
|  |     } | ||||||
| 
 | 
 | ||||||
|     data class Sources( |     private data class ResponseJson( | ||||||
|         @JsonProperty("src") val src: String? = null, |         @JsonProperty("data") val data: ResponseData? | ||||||
|         @JsonProperty("type") val type: String? = null, |     ) | ||||||
|         @JsonProperty("label") val label: String? = null, |     private data class ResponseData( | ||||||
|  |         @JsonProperty("sources") val sources: List<ResponseSources>? = listOf() | ||||||
|  |     ) | ||||||
|  |     private data class ResponseSources( | ||||||
|  |         @JsonProperty("src") val src: String?, | ||||||
|  |         @JsonProperty("type") val type: String?, | ||||||
|  |         @JsonProperty("label") val label: String? | ||||||
|     ) |     ) | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| } | } | ||||||
|  | @ -15,7 +15,7 @@ cloudstream { | ||||||
|     * 2: Slow |     * 2: Slow | ||||||
|     * 3: Beta only |     * 3: Beta only | ||||||
|     * */ |     * */ | ||||||
|     status = 0 // will be 3 if unspecified |     status = 3 // will be 3 if unspecified | ||||||
| 
 | 
 | ||||||
|     // List of video source types. Users are able to filter for extensions in a given category. |     // 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: |     // You can find a list of avaliable types here: | ||||||
|  |  | ||||||
|  | @ -15,7 +15,7 @@ cloudstream { | ||||||
|     * 2: Slow |     * 2: Slow | ||||||
|     * 3: Beta only |     * 3: Beta only | ||||||
|     * */ |     * */ | ||||||
|     status = 0 // will be 3 if unspecified |     status = 1 // will be 3 if unspecified | ||||||
| 
 | 
 | ||||||
|     // List of video source types. Users are able to filter for extensions in a given category. |     // 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: |     // You can find a list of avaliable types here: | ||||||
|  |  | ||||||
|  | @ -15,7 +15,7 @@ cloudstream { | ||||||
|     * 2: Slow |     * 2: Slow | ||||||
|     * 3: Beta only |     * 3: Beta only | ||||||
|     * */ |     * */ | ||||||
|     status = 0 // will be 3 if unspecified |     status = 3 // will be 3 if unspecified | ||||||
| 
 | 
 | ||||||
|     // List of video source types. Users are able to filter for extensions in a given category. |     // 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: |     // You can find a list of avaliable types here: | ||||||
|  |  | ||||||
|  | @ -15,7 +15,7 @@ cloudstream { | ||||||
|     * 2: Slow |     * 2: Slow | ||||||
|     * 3: Beta only |     * 3: Beta only | ||||||
|     * */ |     * */ | ||||||
|     status = 0 // will be 3 if unspecified |     status = 1 // will be 3 if unspecified | ||||||
| 
 | 
 | ||||||
|     // List of video source types. Users are able to filter for extensions in a given category. |     // 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: |     // You can find a list of avaliable types here: | ||||||
|  |  | ||||||
|  | @ -15,7 +15,7 @@ cloudstream { | ||||||
|     * 2: Slow |     * 2: Slow | ||||||
|     * 3: Beta only |     * 3: Beta only | ||||||
|     * */ |     * */ | ||||||
|     status = 0 // will be 3 if unspecified |     status = 3 // will be 3 if unspecified | ||||||
| 
 | 
 | ||||||
|     // List of video source types. Users are able to filter for extensions in a given category. |     // 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: |     // You can find a list of avaliable types here: | ||||||
|  |  | ||||||
|  | @ -15,7 +15,7 @@ cloudstream { | ||||||
|     * 2: Slow |     * 2: Slow | ||||||
|     * 3: Beta only |     * 3: Beta only | ||||||
|     * */ |     * */ | ||||||
|     status = 0 // will be 3 if unspecified |     status = 1 // will be 3 if unspecified | ||||||
| 
 | 
 | ||||||
|     // List of video source types. Users are able to filter for extensions in a given category. |     // 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: |     // You can find a list of avaliable types here: | ||||||
|  |  | ||||||
|  | @ -15,7 +15,7 @@ cloudstream { | ||||||
|     * 2: Slow |     * 2: Slow | ||||||
|     * 3: Beta only |     * 3: Beta only | ||||||
|     * */ |     * */ | ||||||
|     status = 0 // will be 3 if unspecified |     status = 1 // will be 3 if unspecified | ||||||
| 
 | 
 | ||||||
|     // List of video source types. Users are able to filter for extensions in a given category. |     // 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: |     // You can find a list of avaliable types here: | ||||||
|  |  | ||||||
|  | @ -15,7 +15,7 @@ cloudstream { | ||||||
|     * 2: Slow |     * 2: Slow | ||||||
|     * 3: Beta only |     * 3: Beta only | ||||||
|     * */ |     * */ | ||||||
|     status = 0 // will be 3 if unspecified |     status = 2 // will be 3 if unspecified | ||||||
| 
 | 
 | ||||||
|     // List of video source types. Users are able to filter for extensions in a given category. |     // 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: |     // You can find a list of avaliable types here: | ||||||
|  |  | ||||||
|  | @ -1,28 +0,0 @@ | ||||||
| // use an integer for version numbers |  | ||||||
| version = 5 |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| cloudstream { |  | ||||||
|     // All of these properties are optional, you can safely remove them |  | ||||||
| 
 |  | ||||||
|     description = "Pornhits" |  | ||||||
|     authors = listOf("KillerDogeEmpire, Coxju") |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|     * 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("NSFW") |  | ||||||
| 
 |  | ||||||
|     iconUrl = "https://www.google.com/s2/favicons?domain=pornhits.com&sz=%size%" |  | ||||||
| 
 |  | ||||||
|     language = "en" |  | ||||||
| } |  | ||||||
|  | @ -1,2 +0,0 @@ | ||||||
| <?xml version="1.0" encoding="utf-8"?> |  | ||||||
| <manifest package="com.KillerDogeEmpire"/> |  | ||||||
|  | @ -1,216 +0,0 @@ | ||||||
| package com.KillerDogeEmpire |  | ||||||
| 
 |  | ||||||
| import com.lagradost.cloudstream3.HomePageList |  | ||||||
| import com.lagradost.cloudstream3.HomePageResponse |  | ||||||
| import com.lagradost.cloudstream3.LoadResponse |  | ||||||
| import com.lagradost.cloudstream3.MainAPI |  | ||||||
| import com.lagradost.cloudstream3.MainPageRequest |  | ||||||
| import com.lagradost.cloudstream3.SearchResponse |  | ||||||
| import com.lagradost.cloudstream3.SubtitleFile |  | ||||||
| import com.lagradost.cloudstream3.TvType |  | ||||||
| import com.lagradost.cloudstream3.VPNStatus |  | ||||||
| import com.lagradost.cloudstream3.app |  | ||||||
| import com.lagradost.cloudstream3.fixUrl |  | ||||||
| import com.lagradost.cloudstream3.fixUrlNull |  | ||||||
| import com.lagradost.cloudstream3.mainPageOf |  | ||||||
| import com.lagradost.cloudstream3.newHomePageResponse |  | ||||||
| import com.lagradost.cloudstream3.newMovieLoadResponse |  | ||||||
| import com.lagradost.cloudstream3.newMovieSearchResponse |  | ||||||
| import com.lagradost.cloudstream3.utils.ExtractorLink |  | ||||||
| import com.lagradost.cloudstream3.utils.Qualities |  | ||||||
| import org.json.JSONObject |  | ||||||
| import org.jsoup.nodes.Element |  | ||||||
| 
 |  | ||||||
| class Pornhits : MainAPI() { |  | ||||||
|     override var mainUrl = "https://www.pornhits.com" |  | ||||||
|     override var name = "Pornhits" |  | ||||||
|     override val hasMainPage = true |  | ||||||
|     override val hasDownloadSupport = true |  | ||||||
|     override val vpnStatus = VPNStatus.MightBeNeeded |  | ||||||
|     override val supportedTypes = setOf(TvType.NSFW) |  | ||||||
| 
 |  | ||||||
|     override val mainPage = mainPageOf( |  | ||||||
|         "$mainUrl/videos.php?p=%d&s=l" to "Latest", |  | ||||||
|         "$mainUrl/videos.php?p=%d&s=pd" to "Popular last day", |  | ||||||
|         "$mainUrl/videos.php?p=%d&s=bd" to "Top Rated (day)", |  | ||||||
|         "$mainUrl/videos.php?p=%d&s=pw" to "Popular last week", |  | ||||||
|         "$mainUrl/videos.php?p=%d&s=bw" to "Top Rated (week)", |  | ||||||
|         "$mainUrl/videos.php?p=%d&s=pm" to "Popular last month", |  | ||||||
|         "$mainUrl/videos.php?p=%d&s=bm" to "Top Rated (month)", |  | ||||||
|     ) |  | ||||||
| 
 |  | ||||||
|     override suspend fun getMainPage( |  | ||||||
|         page: Int, request: MainPageRequest |  | ||||||
|     ): HomePageResponse { |  | ||||||
|         val document = app.get(request.data.format(page)).document |  | ||||||
|         val home = |  | ||||||
|             document.select("div.main-content section.main-container div.list-videos article.item") |  | ||||||
|                 .mapNotNull { |  | ||||||
|                     it.toSearchResult() |  | ||||||
|                 } |  | ||||||
|         return newHomePageResponse( |  | ||||||
|             list = HomePageList( |  | ||||||
|                 name = request.name, list = home, isHorizontalImages = true |  | ||||||
|             ), hasNext = true |  | ||||||
|         ) |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private fun Element.toSearchResult(): SearchResponse? { |  | ||||||
|         val title = this.selectFirst("div.item-info h2.title")?.text() ?: return null |  | ||||||
|         val href = fixUrl(this.selectFirst("a")!!.attr("href")) |  | ||||||
|         val posterUrl = fixUrlNull(this.select("a div.img img").attr("data-original")) |  | ||||||
|         return newMovieSearchResponse(title, href, TvType.Movie) { |  | ||||||
|             this.posterUrl = posterUrl |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     override suspend fun search(query: String): List<SearchResponse> { |  | ||||||
|         val searchResponse = mutableListOf<SearchResponse>() |  | ||||||
|         for (i in 1..15) { |  | ||||||
|             val document = app.get( |  | ||||||
|                 "$mainUrl/videos.php?p=${i}&q=${query.trim().replace(" ", "+")}" |  | ||||||
|             ).document |  | ||||||
|             val results = |  | ||||||
|                 document.select("div.main-content section.main-container div.list-videos article.item") |  | ||||||
|                     .mapNotNull { |  | ||||||
|                         it.toSearchResult() |  | ||||||
|                     } |  | ||||||
|             searchResponse.addAll(results) |  | ||||||
|             if (results.isEmpty()) break |  | ||||||
|         } |  | ||||||
|         return searchResponse |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     override suspend fun load(url: String): LoadResponse { |  | ||||||
|         val document = app.get(url).document |  | ||||||
| 
 |  | ||||||
|         val title = |  | ||||||
|             document.selectFirst("section.video-holder div.video-info div.info-holder article#tab_video_info.tab-content div.headline h1") |  | ||||||
|                 ?.text() |  | ||||||
|                 ?: "" |  | ||||||
|         val poster = fixUrlNull( |  | ||||||
|             document.selectXpath("//script[contains(text(),'var schemaJson')]").first()?.data() |  | ||||||
|                 ?.replace("\"", "") |  | ||||||
|                 ?.substringAfter("thumbnailUrl:") |  | ||||||
|                 ?.substringBefore(",uploadDate:") |  | ||||||
|                 ?.trim() ?: "" |  | ||||||
|         ) |  | ||||||
|         val tags = |  | ||||||
|             document.select(" section.video-holder div.video-info div.info-holder article#tab_video_info.tab-content div.block-details div.info h3.item a") |  | ||||||
|                 .map { it.text() } |  | ||||||
|         val recommendations = |  | ||||||
|             document.select("div.related-videos div.list-videos article.item") |  | ||||||
|                 .mapNotNull { |  | ||||||
|                     it.toSearchResult() |  | ||||||
|                 } |  | ||||||
| 
 |  | ||||||
|         return newMovieLoadResponse(title, url, TvType.NSFW, url) { |  | ||||||
|             this.posterUrl = poster |  | ||||||
|             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 script = |  | ||||||
|             document.selectXpath("//script[contains(text(),'let vpage_data')]").first()?.html() |  | ||||||
|         var isVHQ = false |  | ||||||
|         if (script != null && script.contains("VHQ")) { |  | ||||||
|             isVHQ = true |  | ||||||
|         } |  | ||||||
|         val pattern = Regex("""window\.initPlayer\((.*])\);""") |  | ||||||
|         val matchResult = pattern.find(script ?: "") |  | ||||||
| 
 |  | ||||||
|         val jsonArray = matchResult?.groups?.get(1)?.value |  | ||||||
| 
 |  | ||||||
|         val encodedString = getEncodedString(jsonArray) ?: "" |  | ||||||
| 
 |  | ||||||
|         val decodedString = customBase64Decoder(encodedString) |  | ||||||
| 
 |  | ||||||
|         val videos = JSONObject("{ videos:$decodedString}").getJSONArray("videos") |  | ||||||
|         val externalLinkList = mutableListOf<ExtractorLink>() |  | ||||||
|         for (i in 0 until videos.length()) { |  | ||||||
|             val video = videos.getJSONObject(i) |  | ||||||
|             var quality = Qualities.Unknown.value |  | ||||||
|             var isM3u8 = false |  | ||||||
|             if (video.getString("format").contains("lq")) { |  | ||||||
|                 quality = Qualities.P480.value |  | ||||||
|             } |  | ||||||
|             if (video.getString("format").contains("hq")) { |  | ||||||
|                 quality = Qualities.P720.value |  | ||||||
|             } |  | ||||||
|             var url = customBase64Decoder(video.getString("video_url")) |  | ||||||
|             if (isVHQ) { |  | ||||||
|                 url = "$url&f=video.m3u8" |  | ||||||
|                 isM3u8 = true |  | ||||||
|                 quality = Qualities.Unknown.value |  | ||||||
|             } |  | ||||||
|             externalLinkList.add( |  | ||||||
|                 ExtractorLink( |  | ||||||
|                     this.name, |  | ||||||
|                     this.name, |  | ||||||
|                     fixUrl(url), |  | ||||||
|                     referer = mainUrl, |  | ||||||
|                     quality = quality, |  | ||||||
|                     isM3u8 = isM3u8 |  | ||||||
|                 ) |  | ||||||
|             ) |  | ||||||
|             if (isVHQ) break |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         externalLinkList.forEach(callback) |  | ||||||
|         return true |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private fun customBase64Decoder(encodedString: String): String { |  | ||||||
|         val base64CharacterSet = "АВСDЕFGHIJKLМNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789.,~" |  | ||||||
|         var decodedString = "" |  | ||||||
|         var currentIndex = 0 |  | ||||||
| 
 |  | ||||||
|         Regex("[^АВСЕМA-Za-z0-9.,~]").find(encodedString)?.let { |  | ||||||
|             println("Error decoding URL") |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         val sanitizedString = encodedString.replace("[^АВСЕМA-Za-z0-9.,~]".toRegex(), "") |  | ||||||
| 
 |  | ||||||
|         do { |  | ||||||
|             val firstCharIndex = base64CharacterSet.indexOf(sanitizedString[currentIndex++]) |  | ||||||
|             val secondCharIndex = base64CharacterSet.indexOf(sanitizedString[currentIndex++]) |  | ||||||
|             val thirdCharIndex = base64CharacterSet.indexOf(sanitizedString[currentIndex++]) |  | ||||||
|             val fourthCharIndex = base64CharacterSet.indexOf(sanitizedString[currentIndex++]) |  | ||||||
| 
 |  | ||||||
|             val reconstructedFirstChar = (firstCharIndex shl 2) or (secondCharIndex shr 4) |  | ||||||
|             val reconstructedSecondChar = ((15 and secondCharIndex) shl 4) or (thirdCharIndex shr 2) |  | ||||||
|             val lastPart = ((3 and thirdCharIndex) shl 6) or fourthCharIndex |  | ||||||
| 
 |  | ||||||
|             decodedString += reconstructedFirstChar.toChar().toString() |  | ||||||
|             if (64 != thirdCharIndex) { |  | ||||||
|                 decodedString += reconstructedSecondChar.toChar().toString() |  | ||||||
|             } |  | ||||||
|             if (64 != fourthCharIndex) { |  | ||||||
|                 decodedString += lastPart.toChar().toString() |  | ||||||
|             } |  | ||||||
|         } while (currentIndex < sanitizedString.length) |  | ||||||
|         return java.net.URLDecoder.decode(decodedString, "UTF-8") |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private fun getEncodedString(json: String?): String? { |  | ||||||
|         val stringPattern = Regex("""'([^']+)',""") |  | ||||||
| 
 |  | ||||||
|         val stringMatch = stringPattern.find(json ?: "") |  | ||||||
| 
 |  | ||||||
|         return when { |  | ||||||
|             stringMatch != null -> stringMatch.groups[1]?.value |  | ||||||
|             else -> null |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
| } |  | ||||||
|  | @ -1,14 +0,0 @@ | ||||||
| package com.KillerDogeEmpire |  | ||||||
| 
 |  | ||||||
| import com.lagradost.cloudstream3.plugins.CloudstreamPlugin |  | ||||||
| import com.lagradost.cloudstream3.plugins.Plugin |  | ||||||
| import android.content.Context |  | ||||||
| import com.KillerDogeEmpire.Pornhits |  | ||||||
| 
 |  | ||||||
| @CloudstreamPlugin |  | ||||||
| class PornhitsProvider: Plugin() { |  | ||||||
|     override fun load(context: Context) { |  | ||||||
|         // All providers should be added in this manner. Please don't edit the providers list directly. |  | ||||||
|         registerMainAPI(Pornhits()) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -1,12 +1,12 @@ | ||||||
| // use an integer for version numbers | // use an integer for version numbers | ||||||
| version = 6 | version = 5 | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| cloudstream { | cloudstream { | ||||||
|     // All of these properties are optional, you can safely remove them |     // All of these properties are optional, you can safely remove them | ||||||
| 
 | 
 | ||||||
|     description = "Cornhub" |     description = "Cornhub" | ||||||
|     authors = listOf("KillerDogeEmpire, Stormunblessed, Jace ,Hexated, Coxju") |     authors = listOf("Stormunblessed", "Jace") | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|     * Status int as the following: |     * Status int as the following: | ||||||
|  |  | ||||||
|  | @ -1,117 +1,142 @@ | ||||||
| package com.jacekun | package com.jacekun | ||||||
| 
 | 
 | ||||||
| import android.util.Log | import com.lagradost.cloudstream3.MainAPI | ||||||
| import org.jsoup.nodes.Element | import com.lagradost.cloudstream3.TvType | ||||||
| import com.lagradost.cloudstream3.* | import com.lagradost.cloudstream3.* | ||||||
|  | import com.lagradost.cloudstream3.mvvm.logError | ||||||
|  | import com.lagradost.cloudstream3.network.WebViewResolver | ||||||
| import com.lagradost.cloudstream3.utils.* | import com.lagradost.cloudstream3.utils.* | ||||||
| import com.lagradost.cloudstream3.LoadResponse.Companion.addActors | import org.jsoup.nodes.Element | ||||||
|  | 
 | ||||||
|  | class Pornhub : MainAPI() { | ||||||
|  |     private val globalTvType = TvType.NSFW | ||||||
| 
 | 
 | ||||||
| class PornHub : MainAPI() { |  | ||||||
|     override var mainUrl = "https://www.pornhub.com" |     override var mainUrl = "https://www.pornhub.com" | ||||||
|     override var name                 = "PornHub" |     override var name = "Pornhub" | ||||||
|     override val hasMainPage = true |     override val hasMainPage = true | ||||||
|     override var lang                 = "en" |  | ||||||
|     override val hasQuickSearch       = false |  | ||||||
|     override val hasDownloadSupport   = true |  | ||||||
|     override val hasChromecastSupport = true |     override val hasChromecastSupport = true | ||||||
|  |     override val hasDownloadSupport = true | ||||||
|  |     override val vpnStatus = VPNStatus.MightBeNeeded //Cause it's a big site | ||||||
|     override val supportedTypes = setOf(TvType.NSFW) |     override val supportedTypes = setOf(TvType.NSFW) | ||||||
|     override val vpnStatus            = VPNStatus.MightBeNeeded |  | ||||||
| 
 | 
 | ||||||
|     override val mainPage = mainPageOf( |     override val mainPage = mainPageOf( | ||||||
|         "${mainUrl}/video?o=mr&hd=1&page="           to "Recently Featured", |         "$mainUrl/video?page=" to "Main Page", | ||||||
|         "${mainUrl}/video?o=tr&t=w&hd=1&page="       to "Top Rated", |  | ||||||
|         "${mainUrl}/video?o=mv&t=w&hd=1&page="       to "Most Viewed", |  | ||||||
|         "${mainUrl}/video?o=ht&t=w&hd=1&page="       to "Hottest", |  | ||||||
|         "${mainUrl}/video?p=professional&hd=1&page=" to "Professional", |  | ||||||
|         "${mainUrl}/video?o=lg&hd=1&page="           to "Longest", |  | ||||||
|         "${mainUrl}/video?p=homemade&hd=1&page="     to "Homemade", |  | ||||||
|         "${mainUrl}/video?o=cm&t=w&hd=1&page="       to "Newest", |  | ||||||
|     ) |     ) | ||||||
| 
 | 
 | ||||||
|     override suspend fun getMainPage(page: Int, request: MainPageRequest): HomePageResponse { |     override suspend fun getMainPage(page: Int, request: MainPageRequest): HomePageResponse { | ||||||
|         val document = app.get(request.data + page).document |         try { | ||||||
|         val home     = document.select("li.pcVideoListItem").mapNotNull { it.toSearchResult() } |             val categoryData = request.data | ||||||
| 
 |             val categoryName = request.name | ||||||
|  |             val pagedLink = if (page > 0) categoryData + page else categoryData | ||||||
|  |             val soup = app.get(pagedLink).document | ||||||
|  |             val home = soup.select("div.sectionWrapper div.wrap").mapNotNull { | ||||||
|  |                 if (it == null) { return@mapNotNull null } | ||||||
|  |                 val title = it.selectFirst("span.title a")?.text() ?: "" | ||||||
|  |                 val link = fixUrlNull(it.selectFirst("a")?.attr("href")) ?: return@mapNotNull null | ||||||
|  |                 val img = fetchImgUrl(it.selectFirst("img")) | ||||||
|  |                 MovieSearchResponse( | ||||||
|  |                     name = title, | ||||||
|  |                     url = link, | ||||||
|  |                     apiName = this.name, | ||||||
|  |                     type = globalTvType, | ||||||
|  |                     posterUrl = img | ||||||
|  |                 ) | ||||||
|  |             } | ||||||
|  |             if (home.isNotEmpty()) { | ||||||
|                 return newHomePageResponse( |                 return newHomePageResponse( | ||||||
|                     list = HomePageList( |                     list = HomePageList( | ||||||
|                 name               = request.name, |                         name = categoryName, | ||||||
|                         list = home, |                         list = home, | ||||||
|                         isHorizontalImages = true |                         isHorizontalImages = true | ||||||
|                     ), |                     ), | ||||||
|                     hasNext = true |                     hasNext = true | ||||||
|                 ) |                 ) | ||||||
|  |             } else { | ||||||
|  |                 throw ErrorLoadingException("No homepage data found!") | ||||||
|             } |             } | ||||||
| 
 |         } catch (e: Exception) { | ||||||
|     private fun Element.toSearchResult(): SearchResponse? { |             //e.printStackTrace() | ||||||
|         val title     = this.selectFirst("a")?.attr("title") ?: return null |             logError(e) | ||||||
|         val link      = this.selectFirst("a")?.attr("href") ?: return null |         } | ||||||
|         val posterUrl = fixUrlNull(this.selectFirst("img.thumb")?.attr("src")) |         throw ErrorLoadingException() | ||||||
| 
 |  | ||||||
|         return newMovieSearchResponse(title, link, TvType.Movie) { this.posterUrl = posterUrl } |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     override suspend fun search(query: String): List<SearchResponse> { |     override suspend fun search(query: String): List<SearchResponse> { | ||||||
|         val document = app.get("${mainUrl}/video/search?search=${query}").document |         val url = "$mainUrl/video/search?search=${query}" | ||||||
| 
 |  | ||||||
|         return document.select("li.pcVideoListItem").mapNotNull { it.toSearchResult() } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     override suspend fun quickSearch(query: String): List<SearchResponse> = search(query) |  | ||||||
| 
 |  | ||||||
|     override suspend fun load(url: String): LoadResponse? { |  | ||||||
|         val document = app.get(url).document |         val document = app.get(url).document | ||||||
| 
 |         return document.select("div.sectionWrapper div.wrap").mapNotNull { | ||||||
|         val title           = document.selectFirst("h1.title span[class='inlineFree']")?.text()?.trim() ?: return null |             if (it == null) { return@mapNotNull null } | ||||||
|         val description     = title |             val title = it.selectFirst("span.title a")?.text() ?: return@mapNotNull null | ||||||
|         val poster          = fixUrlNull(document.selectFirst("div.mainPlayerDiv img")?.attr("src")) |             val link = fixUrlNull(it.selectFirst("a")?.attr("href")) ?: return@mapNotNull null | ||||||
|         val year            = Regex("""uploadDate\": \"(\d+)""").find(document.html())?.groupValues?.get(1)?.toIntOrNull() |             val image = fetchImgUrl(it.selectFirst("img")) | ||||||
|         val tags            = document.select("div.categoriesWrapper a[data-label='Category']").map { it?.text()?.trim().toString().replace(", ","") } |             MovieSearchResponse( | ||||||
|         val rating          = document.selectFirst("span.percent")?.text()?.first()?.toString()?.toRatingInt() |                 name = title, | ||||||
|         val duration        = Regex("duration' : '(.*)',").find(document.html())?.groupValues?.get(1)?.toIntOrNull() |                 url = link, | ||||||
|         val actors          = document.select("div.pornstarsWrapper a[data-label='Pornstar']").mapNotNull { |                 apiName = this.name, | ||||||
|             Actor(it.text().trim(), it.select("img").attr("src")) |                 type = globalTvType, | ||||||
|  |                 posterUrl = image | ||||||
|  |             ) | ||||||
|  |         }.distinctBy { it.url } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|         val recommendations = document.selectXpath("//a[contains(@class, 'img')]").mapNotNull { |     override suspend fun load(url: String): LoadResponse { | ||||||
|             val recName      = it?.attr("title")?.trim() ?: return@mapNotNull null |         val soup = app.get(url).document | ||||||
|             val recHref      = fixUrlNull(it.attr("href")) ?: return@mapNotNull null |         val title = soup.selectFirst(".title span")?.text() ?: "" | ||||||
|             val recPosterUrl = fixUrlNull(it.selectFirst("img")?.attr("src")) |         val poster: String? = soup.selectFirst("div.video-wrapper .mainPlayerDiv img")?.attr("src") ?: | ||||||
|             newMovieSearchResponse(recName, recHref, TvType.NSFW) { |         soup.selectFirst("head meta[property=og:image]")?.attr("content") | ||||||
|                 this.posterUrl = recPosterUrl |         val tags = soup.select("div.categoriesWrapper a") | ||||||
|  |             .map { it?.text()?.trim().toString().replace(", ","") } | ||||||
|  |         return MovieLoadResponse( | ||||||
|  |             name = title, | ||||||
|  |             url = url, | ||||||
|  |             apiName = this.name, | ||||||
|  |             type = globalTvType, | ||||||
|  |             dataUrl = url, | ||||||
|  |             posterUrl = poster, | ||||||
|  |             tags = tags, | ||||||
|  |             plot = title | ||||||
|  |         ) | ||||||
|     } |     } | ||||||
|         } |     override suspend fun loadLinks( | ||||||
| 
 |         data: String, | ||||||
|         return newMovieLoadResponse(title, url, TvType.NSFW, url) { |         isCasting: Boolean, | ||||||
|             this.posterUrl       = poster |         subtitleCallback: (SubtitleFile) -> Unit, | ||||||
|             this.year            = year |         callback: (ExtractorLink) -> Unit | ||||||
|             this.plot            = description |     ): Boolean { | ||||||
|             this.tags            = tags |         app.get( | ||||||
|             this.rating          = rating |             url = data, | ||||||
|             this.duration        = duration |             interceptor = WebViewResolver( | ||||||
|             this.recommendations = recommendations |                 Regex("(master\\.m3u8\\?.*)") | ||||||
|             addActors(actors) |             ) | ||||||
|         } |         ).let { response -> | ||||||
|     } |             M3u8Helper().m3u8Generation( | ||||||
| 
 |                 M3u8Helper.M3u8Stream( | ||||||
|     override suspend fun loadLinks(data: String, isCasting: Boolean, subtitleCallback: (SubtitleFile) -> Unit, callback: (ExtractorLink) -> Unit): Boolean { |                     response.url, | ||||||
|         Log.d("PHub", "url » ${data}") |                     headers = response.headers.toMap() | ||||||
|         val source          = app.get(data).text |                 ), true | ||||||
|         val extracted_value = Regex("""([^\"]*master.m3u8?.[^\"]*)""").find(source)?.groups?.last()?.value ?: return false |             ).apmap { stream -> | ||||||
|         val m3u_link        = extracted_value.replace("\\", "") |                 callback( | ||||||
|         Log.d("PHub", "extracted_value » ${extracted_value}") |  | ||||||
|         Log.d("PHub", "m3u_link » ${m3u_link}") |  | ||||||
| 
 |  | ||||||
|         callback.invoke( |  | ||||||
|                     ExtractorLink( |                     ExtractorLink( | ||||||
|                 source  = this.name, |                         source = name, | ||||||
|                 name    = this.name, |                         name = "${this.name} m3u8", | ||||||
|                 url     = m3u_link, |                         url = stream.streamUrl, | ||||||
|                 referer = "${mainUrl}/", |                         referer = mainUrl, | ||||||
|                 quality = Qualities.Unknown.value, |                         quality = getQualityFromName(stream.quality?.toString()), | ||||||
|                         isM3u8 = true |                         isM3u8 = true | ||||||
|                     ) |                     ) | ||||||
|                 ) |                 ) | ||||||
| 
 |             } | ||||||
|  |         } | ||||||
|         return true |         return true | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|  |     private fun fetchImgUrl(imgsrc: Element?): String? { | ||||||
|  |         return try { imgsrc?.attr("data-src") | ||||||
|  |             ?: imgsrc?.attr("data-mediabook") | ||||||
|  |             ?: imgsrc?.attr("alt") | ||||||
|  |             ?: imgsrc?.attr("data-mediumthumb") | ||||||
|  |             ?: imgsrc?.attr("data-thumb_url") | ||||||
|  |             ?: imgsrc?.attr("src") | ||||||
|  |         } catch (e:Exception) { null } | ||||||
|  |     } | ||||||
| } | } | ||||||
|  | @ -1,28 +0,0 @@ | ||||||
| // use an integer for version numbers |  | ||||||
| version = 5 |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| cloudstream { |  | ||||||
|     // All of these properties are optional, you can safely remove them |  | ||||||
| 
 |  | ||||||
|     description = "Porntrex" |  | ||||||
|     authors = listOf("KillerDogeEmpire, Coxju") |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|     * 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("NSFW") |  | ||||||
| 
 |  | ||||||
|     iconUrl = "https://www.google.com/s2/favicons?domain=www.porntrex.com&sz=%size%" |  | ||||||
| 
 |  | ||||||
|     language = "en" |  | ||||||
| } |  | ||||||
|  | @ -1,2 +0,0 @@ | ||||||
| <?xml version="1.0" encoding="utf-8"?> |  | ||||||
| <manifest package="com.KillerDogeEmpire"/> |  | ||||||
|  | @ -1,183 +0,0 @@ | ||||||
| package com.KillerDogeEmpire |  | ||||||
| 
 |  | ||||||
| import com.lagradost.cloudstream3.HomePageList |  | ||||||
| import com.lagradost.cloudstream3.HomePageResponse |  | ||||||
| import com.lagradost.cloudstream3.LoadResponse |  | ||||||
| import com.lagradost.cloudstream3.MainAPI |  | ||||||
| import com.lagradost.cloudstream3.MainPageRequest |  | ||||||
| import com.lagradost.cloudstream3.SearchResponse |  | ||||||
| import com.lagradost.cloudstream3.SubtitleFile |  | ||||||
| import com.lagradost.cloudstream3.TvType |  | ||||||
| import com.lagradost.cloudstream3.VPNStatus |  | ||||||
| import com.lagradost.cloudstream3.app |  | ||||||
| import com.lagradost.cloudstream3.fixUrl |  | ||||||
| import com.lagradost.cloudstream3.fixUrlNull |  | ||||||
| import com.lagradost.cloudstream3.mainPageOf |  | ||||||
| import com.lagradost.cloudstream3.newHomePageResponse |  | ||||||
| import com.lagradost.cloudstream3.newMovieLoadResponse |  | ||||||
| import com.lagradost.cloudstream3.newMovieSearchResponse |  | ||||||
| import com.lagradost.cloudstream3.utils.ExtractorLink |  | ||||||
| import com.lagradost.cloudstream3.utils.getQualityFromName |  | ||||||
| import org.json.JSONObject |  | ||||||
| import org.jsoup.internal.StringUtil |  | ||||||
| import org.jsoup.nodes.Element |  | ||||||
| 
 |  | ||||||
| class Porntrex : MainAPI() { |  | ||||||
|     override var mainUrl = "https://www.porntrex.com" |  | ||||||
|     override var name = "Porntrex" |  | ||||||
|     override val hasMainPage = true |  | ||||||
|     override val hasDownloadSupport = true |  | ||||||
|     override val vpnStatus = VPNStatus.MightBeNeeded |  | ||||||
|     override val supportedTypes = setOf(TvType.NSFW) |  | ||||||
| 
 |  | ||||||
|     override val mainPage = mainPageOf( |  | ||||||
|             "latest-updates" to "Latest Videos", |  | ||||||
|             "most-popular/daily/?mode=async&function=get_block&block_id=list_videos_common_videos_list_norm&sort_by=video_viewed_today&from4=" to "Most popular daily", |  | ||||||
|             "top-rated/daily/?mode=async&function=get_block&block_id=list_videos_common_videos_list_norm&sort_by=rating_today&from4=" to "Top rated daily", |  | ||||||
|             "most-popular/weekly/?mode=async&function=get_block&block_id=list_videos_common_videos_list_norm&sort_by=video_viewed_week&from4=" to "Most popular weekly", |  | ||||||
|             "top-rated/weekly/?mode=async&function=get_block&block_id=list_videos_common_videos_list_norm&sort_by=rating_week&from4=" to "Top rated weekly", |  | ||||||
|             "most-popular/monthly/?mode=async&function=get_block&block_id=list_videos_common_videos_list_norm&sort_by=video_viewed_month&from4=" to "Most popular monthly", |  | ||||||
|             "top-rated/monthly/?mode=async&function=get_block&block_id=list_videos_common_videos_list_norm&sort_by=rating_month&from4=" to "Top rated monthly", |  | ||||||
|             "most-popular/?mode=async&function=get_block&block_id=list_videos_common_videos_list_norm&sort_by=video_viewed&from4=" to "Most popular all time", |  | ||||||
|             "top-rated/?mode=async&function=get_block&block_id=list_videos_common_videos_list_norm&sort_by=rating&from4=" to "Top rated all time", |  | ||||||
|     ) |  | ||||||
| 
 |  | ||||||
|     override suspend fun getMainPage( |  | ||||||
|             page: Int, |  | ||||||
|             request: MainPageRequest |  | ||||||
|     ): HomePageResponse { |  | ||||||
|         var url: String |  | ||||||
|         url = if (page == 1) { |  | ||||||
|             "$mainUrl/${request.data}/" |  | ||||||
|         } else { |  | ||||||
|             "$mainUrl/${request.data}/${page}/" |  | ||||||
|         } |  | ||||||
|         if (request.data.contains("mode=async")) { |  | ||||||
|             url = "$mainUrl/${request.data}${page}" |  | ||||||
|         } |  | ||||||
|         val document = app.get(url).document |  | ||||||
|         val home = |  | ||||||
|                 document.select("div.video-list div.video-item") |  | ||||||
|                         .mapNotNull { |  | ||||||
|                             it.toSearchResult() |  | ||||||
|                         } |  | ||||||
|         return newHomePageResponse( |  | ||||||
|                 list = HomePageList( |  | ||||||
|                         name = request.name, |  | ||||||
|                         list = home, |  | ||||||
|                         isHorizontalImages = true |  | ||||||
|                 ), |  | ||||||
|                 hasNext = true |  | ||||||
|         ) |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private fun Element.toSearchResult(): SearchResponse? { |  | ||||||
|         val title = this.selectFirst("p.inf a")?.text() ?: return null |  | ||||||
|         val href = fixUrl(this.selectFirst("p.inf a")!!.attr("href")) |  | ||||||
|         val posterUrl = fixUrlNull(this.select("a.thumb img.cover").attr("data-src")) |  | ||||||
|         return newMovieSearchResponse(title, href, TvType.Movie) { |  | ||||||
|             this.posterUrl = posterUrl |  | ||||||
|             this.posterHeaders = mapOf(Pair("referer", "${mainUrl}/")) |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     override suspend fun search(query: String): List<SearchResponse> { |  | ||||||
|         val searchResponse = mutableListOf<SearchResponse>() |  | ||||||
|         for (i in 1..15) { |  | ||||||
|             val url: String = if (i == 1) { |  | ||||||
|                 "$mainUrl/search/${query.replace(" ", "-")}/" |  | ||||||
|             } else { |  | ||||||
|                 "$mainUrl/search/${query.replace(" ", "-")}/$i/" |  | ||||||
|             } |  | ||||||
|             val document = |  | ||||||
|                     app.get(url).document |  | ||||||
|             val results = |  | ||||||
|                     document.select("div.video-list div.video-item") |  | ||||||
|                             .mapNotNull { |  | ||||||
|                                 it.toSearchResult() |  | ||||||
|                             } |  | ||||||
|             searchResponse.addAll(results) |  | ||||||
|             if (results.isEmpty()) break |  | ||||||
|         } |  | ||||||
|         return searchResponse |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     override suspend fun load(url: String): LoadResponse { |  | ||||||
|         val document = app.get(url).document |  | ||||||
| 
 |  | ||||||
|         val jsonObject = JSONObject(document.selectXpath("//script[contains(text(),'var flashvars')]").first()?.data() |  | ||||||
|                 ?.substringAfter("var flashvars = ") |  | ||||||
|                 ?.substringBefore("var player_obj") |  | ||||||
|                 ?.replace(";", "") ?: "") |  | ||||||
| 
 |  | ||||||
|         val title = jsonObject.getString("video_title") |  | ||||||
|         val poster = |  | ||||||
|                 fixUrlNull(jsonObject.getString("preview_url")) |  | ||||||
| 
 |  | ||||||
|         val tags = jsonObject.getString("video_tags").split(", ").map { it.replace("-", "") }.filter { it.isNotBlank() && !StringUtil.isNumeric(it) } |  | ||||||
|         val description = jsonObject.getString("video_title") |  | ||||||
| 
 |  | ||||||
|         val recommendations = |  | ||||||
|                 document.select("div#list_videos_related_videos div.video-list div.video-item").mapNotNull { |  | ||||||
|                     it.toSearchResult() |  | ||||||
|                 } |  | ||||||
| 
 |  | ||||||
|         return newMovieLoadResponse(title, url, TvType.NSFW, url) { |  | ||||||
|             this.posterUrl = poster |  | ||||||
|             this.posterHeaders = mapOf(Pair("referer", "${mainUrl}/")) |  | ||||||
|             this.plot = description |  | ||||||
|             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 jsonObject = JSONObject(document.selectXpath("//script[contains(text(),'var flashvars')]").first()?.data() |  | ||||||
|                 ?.substringAfter("var flashvars = ") |  | ||||||
|                 ?.substringBefore("var player_obj") |  | ||||||
|                 ?.replace(";", "") ?: "") |  | ||||||
|         val extlinkList = mutableListOf<ExtractorLink>() |  | ||||||
|         for (i in 0 until 7) { |  | ||||||
|             var url: String |  | ||||||
|             var quality: String |  | ||||||
|             if (i == 0) { |  | ||||||
|                 url = jsonObject.optString("video_url") ?: "" |  | ||||||
|                 quality = jsonObject.optString("video_url_text") ?: "" |  | ||||||
|             } else { |  | ||||||
|                 if (i == 1) { |  | ||||||
|                     url = jsonObject.optString("video_alt_url") ?: "" |  | ||||||
|                     quality = jsonObject.optString("video_alt_url_text") ?: "" |  | ||||||
|                 } else { |  | ||||||
|                     url = jsonObject.optString("video_alt_url${i}") ?: "" |  | ||||||
|                     quality = jsonObject.optString("video_alt_url${i}_text") ?: "" |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             if (url == "") { |  | ||||||
|                 continue |  | ||||||
|             } |  | ||||||
|             extlinkList.add( |  | ||||||
|                     ExtractorLink( |  | ||||||
|                             name, |  | ||||||
|                             name, |  | ||||||
|                             fixUrl(url), |  | ||||||
|                             referer = "${mainUrl}/", |  | ||||||
|                             quality = |  | ||||||
|                             Regex("(\\d+.)").find(quality)?.groupValues?.get(1) |  | ||||||
|                                     .let { getQualityFromName(it) } |  | ||||||
|                     ) |  | ||||||
|             ) |  | ||||||
|         } |  | ||||||
|         extlinkList.forEach(callback) |  | ||||||
|         return true |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
|  | @ -1,14 +0,0 @@ | ||||||
| package com.KillerDogeEmpire |  | ||||||
| 
 |  | ||||||
| import android.content.Context |  | ||||||
| import com.KillerDogeEmpire.Porntrex |  | ||||||
| import com.lagradost.cloudstream3.plugins.CloudstreamPlugin |  | ||||||
| import com.lagradost.cloudstream3.plugins.Plugin |  | ||||||
| 
 |  | ||||||
| @CloudstreamPlugin |  | ||||||
| class PorntrexProvider : Plugin() { |  | ||||||
|     override fun load(context: Context) { |  | ||||||
|         // All providers should be added in this manner. Please don't edit the providers list directly. |  | ||||||
|         registerMainAPI(Porntrex()) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -1,2 +0,0 @@ | ||||||
| <?xml version="1.0" encoding="utf-8"?> |  | ||||||
| <manifest package="com.KillerDogeEmpire"/> |  | ||||||
|  | @ -1,143 +0,0 @@ | ||||||
| package com.KillerDogeEmpire |  | ||||||
| 
 |  | ||||||
| import com.lagradost.cloudstream3.* |  | ||||||
| import com.lagradost.cloudstream3.utils.AppUtils |  | ||||||
| import com.lagradost.cloudstream3.utils.ExtractorLink |  | ||||||
| import com.lagradost.cloudstream3.utils.Qualities |  | ||||||
| import org.jsoup.nodes.Element |  | ||||||
| 
 |  | ||||||
| class SxyPrn : MainAPI() { |  | ||||||
|     override var mainUrl = "https://sxyprn.com" |  | ||||||
|     override var name = "Sxyprn" |  | ||||||
|     override val hasMainPage = true |  | ||||||
|     override val hasDownloadSupport = true |  | ||||||
|     override val vpnStatus = VPNStatus.MightBeNeeded |  | ||||||
|     override val supportedTypes = setOf(TvType.NSFW) |  | ||||||
| 
 |  | ||||||
|     override val mainPage = mainPageOf( |  | ||||||
|         "$mainUrl/new.html?page=" to "New Videos", |  | ||||||
|         "$mainUrl/new.html?sm=trending&page=" to "Trending", |  | ||||||
|         "$mainUrl/new.html?sm=views&page=" to "Most Viewed", |  | ||||||
|         "$mainUrl/popular/top-viewed.html?p=day" to "Popular - Day", |  | ||||||
|         "$mainUrl/popular/top-viewed.html" to "Popular - Week", |  | ||||||
|         "$mainUrl/popular/top-viewed.html?p=month" to "Popular - Month", |  | ||||||
|         "$mainUrl/popular/top-viewed.html?p=all" to "Popular - All Time" |  | ||||||
|     ) |  | ||||||
| 
 |  | ||||||
|     override suspend fun getMainPage( |  | ||||||
|         page: Int, request: MainPageRequest |  | ||||||
|     ): HomePageResponse { |  | ||||||
|         var pageStr = ((page - 1) * 30).toString() |  | ||||||
| 
 |  | ||||||
|         val document = if ("page=" in request.data) { |  | ||||||
|             app.get(request.data + pageStr).document |  | ||||||
|         } else if ("/blog/" in request.data) { |  | ||||||
|             pageStr = ((page - 1) * 20).toString() |  | ||||||
|             app.get(request.data.replace(".html", "$pageStr.html")).document |  | ||||||
|         } else { |  | ||||||
|             app.get(request.data.replace(".html", ".html/$pageStr")).document |  | ||||||
|         } |  | ||||||
|         val home = document.select("div.main_content div.post_el_small").mapNotNull { |  | ||||||
|                 it.toSearchResult() |  | ||||||
|             } |  | ||||||
|         return newHomePageResponse( |  | ||||||
|             list = HomePageList( |  | ||||||
|                 name = request.name, list = home, isHorizontalImages = true |  | ||||||
|             ), hasNext = true |  | ||||||
|         ) |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private fun Element.toSearchResult(): SearchResponse? { |  | ||||||
|         val title = this.selectFirst("div.post_text")?.text() ?: return null |  | ||||||
|         val href = fixUrl(this.selectFirst("a.js-pop")!!.attr("href")) |  | ||||||
|         var posterUrl = fixUrl(this.select("div.vid_container div.post_vid_thumb img").attr("src")) |  | ||||||
|         if (posterUrl == "") { |  | ||||||
|             posterUrl = |  | ||||||
|                 fixUrl(this.select("div.vid_container div.post_vid_thumb img").attr("data-src")) |  | ||||||
|         } |  | ||||||
|         return newMovieSearchResponse(title, href, TvType.Movie) { |  | ||||||
|             this.posterUrl = posterUrl |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     override suspend fun search(query: String): List<SearchResponse> { |  | ||||||
|         val searchResponse = mutableListOf<SearchResponse>() |  | ||||||
|         for (i in 0 until 15) { |  | ||||||
|             val document = app.get( |  | ||||||
|                 "$mainUrl/${query.replace(" ", "-")}.html?page=${i * 30}" |  | ||||||
|             ).document |  | ||||||
|             val results = document.select("div.main_content div.post_el_small").mapNotNull { |  | ||||||
|                     it.toSearchResult() |  | ||||||
|                 } |  | ||||||
|             if (!searchResponse.containsAll(results)) { |  | ||||||
|                 searchResponse.addAll(results) |  | ||||||
|             } else { |  | ||||||
|                 break |  | ||||||
|             } |  | ||||||
|             if (results.isEmpty()) break |  | ||||||
|         } |  | ||||||
|         return searchResponse |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     override suspend fun load(url: String): LoadResponse { |  | ||||||
|         val document = app.get(url).document |  | ||||||
|         val title = document.selectFirst("div.post_text")?.text()?.trim().toString() |  | ||||||
|         val poster = fixUrlNull( |  | ||||||
|             document.selectFirst("div#vid_container_id meta[itemprop=thumbnailUrl]") |  | ||||||
|                 ?.attr("content") |  | ||||||
|         ) |  | ||||||
| 
 |  | ||||||
|         val recommendations = document.select("div.main_content div div.post_el_small").mapNotNull { |  | ||||||
|             it.toSearchResult() |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         return newMovieLoadResponse(title, url, TvType.NSFW, url) { |  | ||||||
|             this.posterUrl = poster |  | ||||||
|             this.recommendations = recommendations |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private fun updateUrl(arg: MutableList<String>): MutableList<String> { |  | ||||||
|         arg[5] = |  | ||||||
|             (Integer.parseInt(arg[5]) - (generateNumber(arg[6]) + generateNumber(arg[7]))).toString() |  | ||||||
|         return arg |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private fun generateNumber(arg: String): Int { |  | ||||||
|         val str = arg.replace(Regex("\\D"), "") |  | ||||||
|         var sut = 0 |  | ||||||
|         for (element in str) { |  | ||||||
|             sut += Integer.parseInt(element.toString(), 10) |  | ||||||
|         } |  | ||||||
|         return sut |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     override suspend fun loadLinks( |  | ||||||
|         data: String, |  | ||||||
|         isCasting: Boolean, |  | ||||||
|         subtitleCallback: (SubtitleFile) -> Unit, |  | ||||||
|         callback: (ExtractorLink) -> Unit |  | ||||||
|     ): Boolean { |  | ||||||
|         val document = app.get(data).document |  | ||||||
|         val parsed = AppUtils.parseJson<Map<String, String>>( |  | ||||||
|             document.select("span.vidsnfo").attr("data-vnfo") |  | ||||||
|         ) |  | ||||||
|         parsed[parsed.keys.toList()[0]] |  | ||||||
|         var url = parsed[parsed.keys.toList()[0]].toString() |  | ||||||
| 
 |  | ||||||
|         var tmp = url.split("/").toMutableList() |  | ||||||
|         tmp[1] += "8" |  | ||||||
|         tmp = updateUrl(tmp) |  | ||||||
| 
 |  | ||||||
|         url = fixUrl(tmp.joinToString("/")) |  | ||||||
| 
 |  | ||||||
|         callback.invoke( |  | ||||||
|             ExtractorLink( |  | ||||||
|                 this.name, this.name, url, referer = data, quality = Qualities.Unknown.value |  | ||||||
|             ) |  | ||||||
|         ) |  | ||||||
| 
 |  | ||||||
|         return true |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
| } |  | ||||||
|  | @ -1,14 +0,0 @@ | ||||||
| package com.KillerDogeEmpire |  | ||||||
| 
 |  | ||||||
| import android.content.Context |  | ||||||
| import com.KillerDogeEmpire.SxyPrn |  | ||||||
| import com.lagradost.cloudstream3.plugins.CloudstreamPlugin |  | ||||||
| import com.lagradost.cloudstream3.plugins.Plugin |  | ||||||
| 
 |  | ||||||
| @CloudstreamPlugin |  | ||||||
| class SxyPrnProvider : Plugin() { |  | ||||||
|     override fun load(context: Context) { |  | ||||||
|         // All providers should be added in this manner. Please don't edit the providers list directly. |  | ||||||
|         registerMainAPI(SxyPrn()) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -1,28 +0,0 @@ | ||||||
| // use an integer for version numbers |  | ||||||
| version = 5 |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| cloudstream { |  | ||||||
|     // All of these properties are optional, you can safely remove them |  | ||||||
| 
 |  | ||||||
|     description = "Uncutmaza" |  | ||||||
|     authors = listOf("Coxju") |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|     * 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("NSFW") |  | ||||||
| 
 |  | ||||||
|     iconUrl = "https://www.google.com/s2/favicons?domain=uncutmaza.com&sz=%size%" |  | ||||||
| 
 |  | ||||||
|     language = "en" |  | ||||||
| } |  | ||||||
|  | @ -1,2 +0,0 @@ | ||||||
| <?xml version="1.0" encoding="utf-8"?> |  | ||||||
| <manifest package="com.KillerDogeEmpire"/> |  | ||||||
|  | @ -1,118 +0,0 @@ | ||||||
| package com.KillerDogeEmpire |  | ||||||
| 
 |  | ||||||
| import com.lagradost.cloudstream3.HomePageList |  | ||||||
| import com.lagradost.cloudstream3.HomePageResponse |  | ||||||
| import com.lagradost.cloudstream3.LoadResponse |  | ||||||
| import com.lagradost.cloudstream3.MainAPI |  | ||||||
| import com.lagradost.cloudstream3.MainPageRequest |  | ||||||
| import com.lagradost.cloudstream3.SearchResponse |  | ||||||
| import com.lagradost.cloudstream3.SubtitleFile |  | ||||||
| import com.lagradost.cloudstream3.TvType |  | ||||||
| import com.lagradost.cloudstream3.VPNStatus |  | ||||||
| import com.lagradost.cloudstream3.app |  | ||||||
| import com.lagradost.cloudstream3.fixTitle |  | ||||||
| import com.lagradost.cloudstream3.fixUrl |  | ||||||
| import com.lagradost.cloudstream3.fixUrlNull |  | ||||||
| import com.lagradost.cloudstream3.mainPageOf |  | ||||||
| import com.lagradost.cloudstream3.newHomePageResponse |  | ||||||
| import com.lagradost.cloudstream3.newMovieLoadResponse |  | ||||||
| import com.lagradost.cloudstream3.newMovieSearchResponse |  | ||||||
| import com.lagradost.cloudstream3.utils.ExtractorLink |  | ||||||
| import com.lagradost.cloudstream3.utils.Qualities |  | ||||||
| import org.jsoup.nodes.Element |  | ||||||
| 
 |  | ||||||
| class UncutMaza : MainAPI() { |  | ||||||
|     override var mainUrl = "https://uncutmaza.com" |  | ||||||
|     override var name = "Uncutmaza" |  | ||||||
|     override val hasMainPage = true |  | ||||||
|     override val hasDownloadSupport = true |  | ||||||
|     override val vpnStatus = VPNStatus.MightBeNeeded |  | ||||||
|     override val supportedTypes = setOf(TvType.NSFW) |  | ||||||
| 
 |  | ||||||
|     override val mainPage = mainPageOf( |  | ||||||
|             "$mainUrl/page/" to "Home", "$mainUrl/category/niks-indian-porn/page/" to "Niks Indian" |  | ||||||
|     ) |  | ||||||
| 
 |  | ||||||
|     override suspend fun getMainPage( |  | ||||||
|             page: Int, request: MainPageRequest |  | ||||||
|     ): HomePageResponse { |  | ||||||
|         val document = app.get(request.data + page).document |  | ||||||
|         val home = document.select("div.videos-list > article.post").mapNotNull { |  | ||||||
|             it.toSearchResult() |  | ||||||
|         } |  | ||||||
|         return newHomePageResponse( |  | ||||||
|                 list = HomePageList( |  | ||||||
|                         name = request.name, list = home, isHorizontalImages = true |  | ||||||
|                 ), hasNext = true |  | ||||||
|         ) |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private fun Element.toSearchResult(): SearchResponse { |  | ||||||
|         val title = fixTitle(this.select("a").attr("title")) |  | ||||||
|         val href = fixUrl(this.select("a").attr("href")) |  | ||||||
|         val posterUrl = fixUrlNull( |  | ||||||
|                 this.select("a > div.post-thumbnail>div.post-thumbnail-container>img").attr("data-src") |  | ||||||
|         ) |  | ||||||
|         return newMovieSearchResponse(title, href, TvType.Movie) { |  | ||||||
|             this.posterUrl = posterUrl |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     override suspend fun search(query: String): List<SearchResponse> { |  | ||||||
|         val searchResponse = mutableListOf<SearchResponse>() |  | ||||||
|         for (i in 1..5) { |  | ||||||
|             val document = app.get( |  | ||||||
|                     "$mainUrl/page/$i?s=$query" |  | ||||||
|             ).document |  | ||||||
|             val results = document.select("article.post").mapNotNull { |  | ||||||
|                 it.toSearchResult() |  | ||||||
|             } |  | ||||||
|             if (!searchResponse.containsAll(results)) { |  | ||||||
|                 searchResponse.addAll(results) |  | ||||||
|             } else { |  | ||||||
|                 break |  | ||||||
|             } |  | ||||||
|             if (results.isEmpty()) break |  | ||||||
|         } |  | ||||||
|         return searchResponse |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     override suspend fun load(url: String): LoadResponse { |  | ||||||
|         val document = app.get(url).document |  | ||||||
| 
 |  | ||||||
|         val title = |  | ||||||
|                 document.selectFirst("meta[property=og:title]")?.attr("content")?.trim().toString() |  | ||||||
|         val poster = |  | ||||||
|                 fixUrlNull(document.selectFirst("meta[property=og:image]")?.attr("content").toString()) |  | ||||||
|         val description = |  | ||||||
|                 document.selectFirst("meta[property=og:description]")?.attr("content")?.trim() |  | ||||||
| 
 |  | ||||||
|         return newMovieLoadResponse(title, url, TvType.NSFW, url) { |  | ||||||
|             this.posterUrl = poster |  | ||||||
|             this.plot = description |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     override suspend fun loadLinks( |  | ||||||
|             data: String, |  | ||||||
|             isCasting: Boolean, |  | ||||||
|             subtitleCallback: (SubtitleFile) -> Unit, |  | ||||||
|             callback: (ExtractorLink) -> Unit |  | ||||||
|     ): Boolean { |  | ||||||
|         val document = app.get(data).document |  | ||||||
|         document.select("div.video-player").map { res -> |  | ||||||
|             callback.invoke( |  | ||||||
|                     ExtractorLink( |  | ||||||
|                             this.name, this.name, fixUrl( |  | ||||||
|                             res.selectFirst("meta[itemprop=contentURL]")?.attr("content")?.trim() |  | ||||||
|                                     .toString() |  | ||||||
|                     ), referer = data, quality = Qualities.Unknown.value |  | ||||||
|                     ) |  | ||||||
|             ) |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         return true |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
| } |  | ||||||
|  | @ -1,14 +0,0 @@ | ||||||
| package com.KillerDogeEmpire |  | ||||||
| 
 |  | ||||||
| import com.lagradost.cloudstream3.plugins.CloudstreamPlugin |  | ||||||
| import com.lagradost.cloudstream3.plugins.Plugin |  | ||||||
| import android.content.Context |  | ||||||
| import com.KillerDogeEmpire.UncutMaza |  | ||||||
| 
 |  | ||||||
| @CloudstreamPlugin |  | ||||||
| class UncutMazaProvider: Plugin() { |  | ||||||
|     override fun load(context: Context) { |  | ||||||
|         // All providers should be added in this manner. Please don't edit the providers list directly. |  | ||||||
|         registerMainAPI(UncutMaza()) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -15,7 +15,7 @@ cloudstream { | ||||||
|     * 2: Slow |     * 2: Slow | ||||||
|     * 3: Beta only |     * 3: Beta only | ||||||
|     * */ |     * */ | ||||||
|     status = 0 // will be 3 if unspecified |     status = 3 // will be 3 if unspecified | ||||||
| 
 | 
 | ||||||
|     // List of video source types. Users are able to filter for extensions in a given category. |     // 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: |     // You can find a list of avaliable types here: | ||||||
|  |  | ||||||
|  | @ -1,28 +0,0 @@ | ||||||
| // use an integer for version numbers |  | ||||||
| version = 5 |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| cloudstream { |  | ||||||
|     // All of these properties are optional, you can safely remove them |  | ||||||
| 
 |  | ||||||
|     description = "Xhamster" |  | ||||||
|     authors = listOf("KillerDogeEmpire, Coxju") |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|     * 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("NSFW") |  | ||||||
| 
 |  | ||||||
|     iconUrl = "https://www.google.com/s2/favicons?domain=xhamster.com&sz=%size%" |  | ||||||
| 
 |  | ||||||
|     language = "en" |  | ||||||
| } |  | ||||||
|  | @ -1,2 +0,0 @@ | ||||||
| <?xml version="1.0" encoding="utf-8"?> |  | ||||||
| <manifest package="com.KillerDogeEmpire"/> |  | ||||||
|  | @ -1,100 +0,0 @@ | ||||||
| package com.KillerDogeEmpire |  | ||||||
| 
 |  | ||||||
| import com.lagradost.cloudstream3.* |  | ||||||
| import com.lagradost.cloudstream3.utils.ExtractorLink |  | ||||||
| import com.lagradost.cloudstream3.utils.Qualities |  | ||||||
| import org.jsoup.nodes.Element |  | ||||||
| 
 |  | ||||||
| class Xhamster : MainAPI() { |  | ||||||
|     override var mainUrl              = "https://xhamster.com" |  | ||||||
|     override var name                 = "xHamster" |  | ||||||
|     override val hasMainPage          = true |  | ||||||
|     override var lang                 = "en" |  | ||||||
|     override val hasQuickSearch       = false |  | ||||||
|     override val hasDownloadSupport   = true |  | ||||||
|     override val hasChromecastSupport = true |  | ||||||
|     override val supportedTypes       = setOf(TvType.NSFW) |  | ||||||
|     override val vpnStatus            = VPNStatus.MightBeNeeded |  | ||||||
| 
 |  | ||||||
|     override val mainPage = mainPageOf( |  | ||||||
|         "${mainUrl}/newest/"              to "Newest", |  | ||||||
|         "${mainUrl}/most-viewed/weekly/"  to "Most viewed weekly", |  | ||||||
|         "${mainUrl}/most-viewed/monthly/" to "Most viewed monthly", |  | ||||||
|         "${mainUrl}/most-viewed"          to "Most viewed all time", |  | ||||||
|         "${mainUrl}/most-viewed/weekly/"  to "Most viewed weekly" |  | ||||||
|     ) |  | ||||||
| 
 |  | ||||||
|     override suspend fun getMainPage(page: Int, request: MainPageRequest): HomePageResponse { |  | ||||||
|         val document = app.get(request.data + page + "?x_platform_switch=desktop").document |  | ||||||
|         val home     = document.select("div.thumb-list div.thumb-list__item").mapNotNull { it.toSearchResult() } |  | ||||||
| 
 |  | ||||||
|         return newHomePageResponse( |  | ||||||
|             list    = HomePageList( |  | ||||||
|                 name               = request.name, |  | ||||||
|                 list               = home, |  | ||||||
|                 isHorizontalImages = true |  | ||||||
|             ), |  | ||||||
|             hasNext = true |  | ||||||
|         ) |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private fun Element.toSearchResult(): SearchResponse? { |  | ||||||
|         val title     = this.selectFirst("a.video-thumb-info__name")?.text() ?: return null |  | ||||||
|         val href      = fixUrl(this.selectFirst("a.video-thumb-info__name")!!.attr("href")) |  | ||||||
|         val posterUrl = fixUrlNull(this.select("img.thumb-image-container__image").attr("src")) |  | ||||||
| 
 |  | ||||||
|         return newMovieSearchResponse(title, href, TvType.Movie) { this.posterUrl = posterUrl } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     override suspend fun search(query: String): List<SearchResponse> { |  | ||||||
|         val searchResponse = mutableListOf<SearchResponse>() |  | ||||||
| 
 |  | ||||||
|         for (i in 0 until 15) { |  | ||||||
|             val document = app.get("${mainUrl}/search/${query.replace(" ", "+")}/?page=$i&x_platform_switch=desktop").document |  | ||||||
| 
 |  | ||||||
|             val results = document.select("div.thumb-list div.thumb-list__item").mapNotNull { it.toSearchResult() } |  | ||||||
| 
 |  | ||||||
|             if (!searchResponse.containsAll(results)) { |  | ||||||
|                 searchResponse.addAll(results) |  | ||||||
|             } else { |  | ||||||
|                 break |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             if (results.isEmpty()) break |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         return searchResponse |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     override suspend fun load(url: String): LoadResponse { |  | ||||||
|         val document = app.get(url).document |  | ||||||
| 
 |  | ||||||
|         val title           = document.selectFirst("div.with-player-container h1")?.text()?.trim().toString() |  | ||||||
|         val poster          = fixUrlNull(document.selectFirst("div.xp-preload-image")?.attr("style")?.substringAfter("https:")?.substringBefore("\');")) |  | ||||||
|         val tags            = document.select(" nav#video-tags-list-container ul.root-8199e.video-categories-tags.collapsed-8199e li.item-8199e a.video-tag").map { it.text() } |  | ||||||
|         val recommendations = document.select("div.related-container div.thumb-list div.thumb-list__item").mapNotNull { it.toSearchResult() } |  | ||||||
| 
 |  | ||||||
|         return newMovieLoadResponse(title, url, TvType.NSFW, url) { |  | ||||||
|             this.posterUrl       = poster |  | ||||||
|             this.tags            = tags |  | ||||||
|             this.recommendations = recommendations |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     override suspend fun loadLinks(data: String, isCasting: Boolean, subtitleCallback: (SubtitleFile) -> Unit, callback: (ExtractorLink) -> Unit): Boolean { |  | ||||||
|         app.get(url = data).let { response -> |  | ||||||
|             callback( |  | ||||||
|                 ExtractorLink( |  | ||||||
|                     source  = name, |  | ||||||
|                     name    = name, |  | ||||||
|                     url     = fixUrl(response.document.selectXpath("//link[contains(@href,'.m3u8')]")[0]?.attr("href").toString()), |  | ||||||
|                     referer = mainUrl, |  | ||||||
|                     quality = Qualities.Unknown.value, |  | ||||||
|                     isM3u8  = true |  | ||||||
|                 ) |  | ||||||
|             ) |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         return true |  | ||||||
|     } |  | ||||||
| } |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue