mirror of
				https://github.com/recloudstream/cloudstream-extensions.git
				synced 2024-08-15 03:03:54 +00:00 
			
		
		
		
	Merge branch 'recloudstream:master' into master
This commit is contained in:
		
						commit
						0be3872afa
					
				
					 13 changed files with 313 additions and 118 deletions
				
			
		|  | @ -1,5 +1,5 @@ | ||||||
| // use an integer for version numbers | // use an integer for version numbers | ||||||
| version = 3 | version = 4 | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| cloudstream { | cloudstream { | ||||||
|  |  | ||||||
|  | @ -3,7 +3,9 @@ package com.lagradost | ||||||
| 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.addActors | import com.lagradost.cloudstream3.LoadResponse.Companion.addActors | ||||||
|  | import com.lagradost.cloudstream3.LoadResponse.Companion.addTrailer | ||||||
| import com.lagradost.cloudstream3.mvvm.safeApiCall | import com.lagradost.cloudstream3.mvvm.safeApiCall | ||||||
|  | import com.lagradost.cloudstream3.ui.settings.SettingsProviders | ||||||
| import com.lagradost.cloudstream3.utils.AppUtils.parseJson | import com.lagradost.cloudstream3.utils.AppUtils.parseJson | ||||||
| import com.lagradost.cloudstream3.utils.ExtractorLink | import com.lagradost.cloudstream3.utils.ExtractorLink | ||||||
| import com.lagradost.cloudstream3.utils.M3u8Helper | import com.lagradost.cloudstream3.utils.M3u8Helper | ||||||
|  | @ -54,8 +56,13 @@ class AllAnimeProvider : MainAPI() { | ||||||
|         @JsonProperty("availableEpisodes") val availableEpisodes: AvailableEpisodes?, |         @JsonProperty("availableEpisodes") val availableEpisodes: AvailableEpisodes?, | ||||||
|         @JsonProperty("availableEpisodesDetail") val availableEpisodesDetail: AvailableEpisodesDetail?, |         @JsonProperty("availableEpisodesDetail") val availableEpisodesDetail: AvailableEpisodesDetail?, | ||||||
|         @JsonProperty("studios") val studios: List<String>?, |         @JsonProperty("studios") val studios: List<String>?, | ||||||
|  |         @JsonProperty("genres") val genres: List<String>?, | ||||||
|  |         @JsonProperty("averageScore") val averageScore: Int?, | ||||||
|         @JsonProperty("description") val description: String?, |         @JsonProperty("description") val description: String?, | ||||||
|         @JsonProperty("status") val status: String?, |         @JsonProperty("status") val status: String?, | ||||||
|  |         @JsonProperty("banner") val banner : String?, | ||||||
|  |         @JsonProperty("episodeDuration") val episodeDuration : Int?, | ||||||
|  |         @JsonProperty("prevideos") val prevideos : List<String> = emptyList(), | ||||||
|         ) |         ) | ||||||
| 
 | 
 | ||||||
|     private data class AvailableEpisodes( |     private data class AvailableEpisodes( | ||||||
|  | @ -104,56 +111,70 @@ class AllAnimeProvider : MainAPI() { | ||||||
|         @JsonProperty("__typename") val _typename: String? = null |         @JsonProperty("__typename") val _typename: String? = null | ||||||
|     ) |     ) | ||||||
| 
 | 
 | ||||||
|     override suspend fun getMainPage(page: Int, request : MainPageRequest): HomePageResponse { |     private val popularTitle = "Popular" | ||||||
|         val items = ArrayList<HomePageList>() |     private val recentTitle = "Recently updated" | ||||||
|         val urls = listOf( |     override val mainPage = listOf( | ||||||
| //            Pair( |         MainPageData( | ||||||
| //                "Top Anime", |             recentTitle, | ||||||
| //                """$mainUrl/graphql?variables={"type":"anime","size":30,"dateRange":30}&extensions={"persistedQuery":{"version":1,"sha256Hash":"276d52ba09ca48ce2b8beb3affb26d9d673b22f9d1fd4892aaa39524128bc745"}}""" |             """$mainUrl/allanimeapi?variables={"search":{"sortBy":"Recent","allowAdult":${settingsForProvider.enableAdult},"allowUnknown":false},"limit":26,"page":%d,"translationType":"sub","countryOrigin":"ALL"}&extensions={"persistedQuery":{"version":1,"sha256Hash":"9c7a8bc1e095a34f2972699e8105f7aaf9082c6e1ccd56eab99c2f1a971152c6"}}""" | ||||||
| //            ), |  | ||||||
|             // "countryOrigin":"JP" for Japanese only |  | ||||||
|             Pair( |  | ||||||
|                 "Recently updated", |  | ||||||
|                 """$mainUrl/graphql?variables={"search":{"allowAdult":false,"allowUnknown":false},"limit":30,"page":1,"translationType":"dub","countryOrigin":"ALL"}&extensions={"persistedQuery":{"version":1,"sha256Hash":"d2670e3e27ee109630991152c8484fce5ff5e280c523378001f9a23dc1839068"}}""" |  | ||||||
|         ), |         ), | ||||||
|  |         MainPageData( | ||||||
|  |             popularTitle, | ||||||
|  |             """$mainUrl/allanimeapi?variables={"type":"anime","size":30,"dateRange":1,"page":%d,"allowAdult":${settingsForProvider.enableAdult},"allowUnknown":false}&extensions={"persistedQuery":{"version":1,"sha256Hash":"6f6fe5663e3e9ea60bdfa693f878499badab83e7f18b56acdba5f8e8662002aa"}}""" | ||||||
|  |         ) | ||||||
|     ) |     ) | ||||||
| 
 | 
 | ||||||
|         val random = |     override suspend fun getMainPage(page: Int, request: MainPageRequest): HomePageResponse { | ||||||
|             """$mainUrl/graphql?variables={"format":"anime"}&extensions={"persistedQuery":{"version":1,"sha256Hash":"21ac672633498a3698e8f6a93ce6c2b3722b29a216dcca93363bf012c360cd54"}}""" |         val url = request.data.format(page) | ||||||
|         val ranlink = app.get(random).text |  | ||||||
|         val jsonran = parseJson<RandomMain>(ranlink) |  | ||||||
|         val ranhome = jsonran.data?.queryRandomRecommendation?.map { |  | ||||||
|             newAnimeSearchResponse(it.name!!, "$mainUrl/anime/${it.Id}", fix = false) { |  | ||||||
|                 this.posterUrl = it.thumbnail |  | ||||||
|                 this.otherName = it.nativeName |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         items.add(HomePageList("Random", ranhome!!)) |  | ||||||
| 
 |  | ||||||
|         urls.apmap { (HomeName, url) -> |  | ||||||
|         val test = app.get(url).text |         val test = app.get(url).text | ||||||
|  | 
 | ||||||
|  |         val home = when (request.name) { | ||||||
|  |             recentTitle -> { | ||||||
|                 val json = parseJson<AllAnimeQuery>(test) |                 val json = parseJson<AllAnimeQuery>(test) | ||||||
|             val home = ArrayList<SearchResponse>() |  | ||||||
|                 val results = json.data.shows.edges.filter { |                 val results = json.data.shows.edges.filter { | ||||||
|                     // filtering in case there is an anime with 0 episodes available on the site. |                     // filtering in case there is an anime with 0 episodes available on the site. | ||||||
|                     !(it.availableEpisodes?.raw == 0 && it.availableEpisodes.sub == 0 && it.availableEpisodes.dub == 0) |                     !(it.availableEpisodes?.raw == 0 && it.availableEpisodes.sub == 0 && it.availableEpisodes.dub == 0) | ||||||
|                 } |                 } | ||||||
|  | 
 | ||||||
|                 results.map { |                 results.map { | ||||||
|                 home.add( |  | ||||||
|                     newAnimeSearchResponse(it.name, "$mainUrl/anime/${it.Id}", fix = false) { |                     newAnimeSearchResponse(it.name, "$mainUrl/anime/${it.Id}", fix = false) { | ||||||
|                         this.posterUrl = it.thumbnail |                         this.posterUrl = it.thumbnail | ||||||
|                         this.year = it.airedStart?.year |                         this.year = it.airedStart?.year | ||||||
|                         this.otherName = it.englishName |                         this.otherName = it.englishName | ||||||
|                         addDub(it.availableEpisodes?.dub) |                         addDub(it.availableEpisodes?.dub) | ||||||
|                         addSub(it.availableEpisodes?.sub) |                         addSub(it.availableEpisodes?.sub) | ||||||
|                     }) |  | ||||||
|                     } |                     } | ||||||
|             items.add(HomePageList(HomeName, home)) |                 } | ||||||
|  |             } | ||||||
|  |             popularTitle -> { | ||||||
|  |                 val json = parseJson<PopularQuery>(test) | ||||||
|  |                 val results = json.data?.queryPopular?.recommendations?.filter { | ||||||
|  |                     // filtering in case there is an anime with 0 episodes available on the site. | ||||||
|  |                     !(it.anyCard?.availableEpisodes?.raw == 0 && it.anyCard.availableEpisodes.sub == 0 && it.anyCard.availableEpisodes.dub == 0) | ||||||
|  |                 } | ||||||
|  |                 results?.mapNotNull { | ||||||
|  |                     newAnimeSearchResponse( | ||||||
|  |                         it.anyCard?.name ?: return@mapNotNull null, | ||||||
|  |                         "$mainUrl/anime/${it.anyCard.Id ?: it.pageStatus?.Id}", | ||||||
|  |                         fix = false | ||||||
|  |                     ) { | ||||||
|  |                         this.posterUrl = it.anyCard.thumbnail | ||||||
|  |                         this.otherName = it.anyCard.englishName | ||||||
|  |                         addDub(it.anyCard.availableEpisodes?.dub) | ||||||
|  |                         addSub(it.anyCard.availableEpisodes?.sub) | ||||||
|  |                     } | ||||||
|  |                 } ?: emptyList() | ||||||
|  |             } | ||||||
|  |             else -> emptyList() | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         if (items.size <= 0) throw ErrorLoadingException() | 
 | ||||||
|         return HomePageResponse(items) | 
 | ||||||
|  |         return HomePageResponse( | ||||||
|  |             listOf( | ||||||
|  |                 HomePageList(request.name, home) | ||||||
|  |             ), hasNext = home.isNotEmpty() | ||||||
|  |         ) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     override suspend fun search(query: String): List<SearchResponse> { |     override suspend fun search(query: String): List<SearchResponse> { | ||||||
|  | @ -210,6 +231,7 @@ class AllAnimeProvider : MainAPI() { | ||||||
| 
 | 
 | ||||||
|         rhino.evaluateString(scope, js, "JavaScript", 1, null) |         rhino.evaluateString(scope, js, "JavaScript", 1, null) | ||||||
|         val jsEval = scope.get("returnValue", scope) ?: return null |         val jsEval = scope.get("returnValue", scope) ?: return null | ||||||
|  | 
 | ||||||
|         val showData = parseJson<Edges>(jsEval as String) |         val showData = parseJson<Edges>(jsEval as String) | ||||||
| 
 | 
 | ||||||
|         val title = showData.name |         val title = showData.name | ||||||
|  | @ -241,7 +263,7 @@ class AllAnimeProvider : MainAPI() { | ||||||
|             Pair(Actor(name, img), role) |             Pair(Actor(name, img), role) | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         // bruh, they use graphql |         // bruh, they use graphql and bruh it is fucked | ||||||
|         //val recommendations = soup.select("#suggesction > div > div.p > .swipercard")?.mapNotNull { |         //val recommendations = soup.select("#suggesction > div > div.p > .swipercard")?.mapNotNull { | ||||||
|         //    val recTitle = it?.selectFirst(".showname > a") ?: return@mapNotNull null |         //    val recTitle = it?.selectFirst(".showname > a") ?: return@mapNotNull null | ||||||
|         //    val recName = recTitle.text() ?: return@mapNotNull null |         //    val recName = recTitle.text() ?: return@mapNotNull null | ||||||
|  | @ -252,7 +274,12 @@ class AllAnimeProvider : MainAPI() { | ||||||
| 
 | 
 | ||||||
|         return newAnimeLoadResponse(title, url, TvType.Anime) { |         return newAnimeLoadResponse(title, url, TvType.Anime) { | ||||||
|             posterUrl = poster |             posterUrl = poster | ||||||
|  |             backgroundPosterUrl = showData.banner | ||||||
|  |             rating = showData.averageScore?.times(100) | ||||||
|  |             tags = showData.genres | ||||||
|             year = showData.airedStart?.year |             year = showData.airedStart?.year | ||||||
|  |             duration = showData.episodeDuration?.div(60_000) | ||||||
|  |             addTrailer(showData.prevideos.filter { it.isNotBlank() }.map { "https://www.youtube.com/watch?v=$it" }) | ||||||
| 
 | 
 | ||||||
|             addEpisodes(DubStatus.Subbed, episodes.first) |             addEpisodes(DubStatus.Subbed, episodes.first) | ||||||
|             addEpisodes(DubStatus.Dubbed, episodes.second) |             addEpisodes(DubStatus.Dubbed, episodes.second) | ||||||
|  |  | ||||||
|  | @ -0,0 +1,68 @@ | ||||||
|  | package com.lagradost | ||||||
|  | 
 | ||||||
|  | import com.fasterxml.jackson.annotation.JsonProperty | ||||||
|  | 
 | ||||||
|  | data class PopularQuery( | ||||||
|  |     @JsonProperty("data") val data: Data? = Data() | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | data class AvailableEpisodes( | ||||||
|  |     @JsonProperty("sub") val sub: Int? = null, | ||||||
|  |     @JsonProperty("dub") val dub: Int? = null, | ||||||
|  |     @JsonProperty("raw") val raw: Int? = null | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | data class Sub( | ||||||
|  |     @JsonProperty("hour") val hour: Int? = null, | ||||||
|  |     @JsonProperty("minute") val minute: Int? = null, | ||||||
|  |     @JsonProperty("year") val year: Int? = null, | ||||||
|  |     @JsonProperty("month") val month: Int? = null, | ||||||
|  |     @JsonProperty("date") val date: Int? = null | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | data class LastEpisodeDate( | ||||||
|  |     @JsonProperty("dub") val dub: Sub? = Sub(), | ||||||
|  |     @JsonProperty("sub") val sub: Sub? = Sub(), | ||||||
|  |     @JsonProperty("raw") val raw: Sub? = Sub() | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | data class AnyCard( | ||||||
|  |     @JsonProperty("_id") val Id: String? = null, | ||||||
|  |     @JsonProperty("name") val name: String? = null, | ||||||
|  |     @JsonProperty("englishName") val englishName: String? = null, | ||||||
|  |     @JsonProperty("nativeName") val nativeName: String? = null, | ||||||
|  |     @JsonProperty("availableEpisodes") val availableEpisodes: AvailableEpisodes? = AvailableEpisodes(), | ||||||
|  |     @JsonProperty("score") val score: Double? = null, | ||||||
|  |     @JsonProperty("lastEpisodeDate") val lastEpisodeDate: LastEpisodeDate? = LastEpisodeDate(), | ||||||
|  |     @JsonProperty("thumbnail") val thumbnail: String? = null, | ||||||
|  |     @JsonProperty("lastChapterDate") val lastChapterDate: String? = null, | ||||||
|  |     @JsonProperty("availableChapters") val availableChapters: String? = null, | ||||||
|  |     @JsonProperty("__typename") val _typename: String? = null | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | data class PageStatus( | ||||||
|  |     @JsonProperty("_id") val Id: String? = null, | ||||||
|  |     @JsonProperty("views") val views: String? = null, | ||||||
|  |     @JsonProperty("showId") val showId: String? = null, | ||||||
|  |     @JsonProperty("rangeViews") val rangeViews: String? = null, | ||||||
|  |     @JsonProperty("isManga") val isManga: Boolean? = null, | ||||||
|  |     @JsonProperty("__typename") val _typename: String? = null | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | data class Recommendations( | ||||||
|  |     @JsonProperty("anyCard") val anyCard: AnyCard? = AnyCard(), | ||||||
|  |     @JsonProperty("pageStatus") val pageStatus: PageStatus? = PageStatus(), | ||||||
|  |     @JsonProperty("__typename") val _typename: String? = null | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | data class QueryPopular( | ||||||
|  |     @JsonProperty("total") val total: Int? = null, | ||||||
|  |     @JsonProperty("recommendations") val recommendations: ArrayList<Recommendations> = arrayListOf(), | ||||||
|  |     @JsonProperty("__typename") val _typename: String? = null | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | data class Data( | ||||||
|  |     @JsonProperty("queryPopular") val queryPopular: QueryPopular? = QueryPopular() | ||||||
|  | ) | ||||||
|  | @ -16,7 +16,7 @@ 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 | ||||||
|     tvTypes = listOf("AnimeMovie", "Anime", "OVA") |     tvTypes = listOf("AnimeMovie", "Anime", "OVA") | ||||||
|     iconUrl = "https://www.google.com/s2/favicons?domain=crunchyroll.com&sz=%size%" |     iconUrl = "https://www.google.com/s2/favicons?domain=crunchyroll.com&sz=%size%" | ||||||
| } | } | ||||||
|  | @ -1,5 +1,5 @@ | ||||||
| // use an integer for version numbers | // use an integer for version numbers | ||||||
| version = 9 | version = 11 | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| cloudstream { | cloudstream { | ||||||
|  |  | ||||||
|  | @ -458,11 +458,8 @@ open class SflixProvider : MainAPI() { | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         suspend fun getKey(): String? { |         suspend fun getKey(): String? { | ||||||
|             data class KeyObject( |             return app.get("https://raw.githubusercontent.com/consumet/rapidclown/rabbitstream/key.txt") | ||||||
|                 @JsonProperty("key") val key: String? = null |                 .text | ||||||
|             ) |  | ||||||
|             return app.get("https://raw.githubusercontent.com/BlipBlob/blabflow/main/keys.json") |  | ||||||
|                 .parsed<KeyObject>().key |  | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         /** |         /** | ||||||
|  |  | ||||||
|  | @ -1,6 +1,7 @@ | ||||||
| package com.lagradost | package com.lagradost | ||||||
| 
 | 
 | ||||||
| import com.fasterxml.jackson.annotation.JsonProperty | import com.fasterxml.jackson.annotation.JsonProperty | ||||||
|  | import com.lagradost.SflixProvider.Companion.extractRabbitStream | ||||||
| import com.lagradost.cloudstream3.* | import com.lagradost.cloudstream3.* | ||||||
| import com.lagradost.cloudstream3.LoadResponse.Companion.addAniListId | import com.lagradost.cloudstream3.LoadResponse.Companion.addAniListId | ||||||
| import com.lagradost.cloudstream3.LoadResponse.Companion.addMalId | import com.lagradost.cloudstream3.LoadResponse.Companion.addMalId | ||||||
|  | @ -345,21 +346,21 @@ class ZoroProvider : MainAPI() { | ||||||
|             val extractorLink = app.get( |             val extractorLink = app.get( | ||||||
|                 link, |                 link, | ||||||
|             ).parsed<RapidCloudResponse>().link |             ).parsed<RapidCloudResponse>().link | ||||||
| //            val hasLoadedExtractorLink = |             val hasLoadedExtractorLink = | ||||||
|                 loadExtractor(extractorLink, "https://rapid-cloud.ru/", subtitleCallback, callback) |                 loadExtractor(extractorLink, "https://rapid-cloud.ru/", subtitleCallback, callback) | ||||||
| //            if (!hasLoadedExtractorLink) { |             if (!hasLoadedExtractorLink) { | ||||||
| //                extractRabbitStream( |                 extractRabbitStream( | ||||||
| //                    extractorLink, |                     extractorLink, | ||||||
| //                    subtitleCallback, |                     subtitleCallback, | ||||||
| //                    // Blacklist VidCloud for now |                     // Blacklist VidCloud for now | ||||||
| //                    { videoLink -> if (!videoLink.url.contains("betterstream")) callback(videoLink) }, |                     { videoLink -> if (!videoLink.url.contains("betterstream")) callback(videoLink) }, | ||||||
| ////                    false, |                     false, | ||||||
| ////                    extractorData, |                     null, | ||||||
| ////                    decryptKey = getKey() |                     decryptKey = getKey() | ||||||
| //                ) { sourceName -> |                 ) { sourceName -> | ||||||
| //                    sourceName + " - ${it.first}" |                     sourceName + " - ${it.first}" | ||||||
| //                } |                 } | ||||||
| //            } |             } | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         return true |         return true | ||||||
|  |  | ||||||
|  | @ -13,6 +13,8 @@ import com.lagradost.cloudstream3.utils.loadExtractor | ||||||
| import org.json.JSONObject | import org.json.JSONObject | ||||||
| import java.net.URLEncoder | import java.net.URLEncoder | ||||||
| 
 | 
 | ||||||
|  | private const val TRACKER_LIST_URL = "https://raw.githubusercontent.com/ngosang/trackerslist/master/trackers_best.txt" | ||||||
|  | 
 | ||||||
| class StremioProvider : MainAPI() { | class StremioProvider : MainAPI() { | ||||||
|     override var mainUrl = "https://stremio.github.io/stremio-static-addon-example" |     override var mainUrl = "https://stremio.github.io/stremio-static-addon-example" | ||||||
|     override var name = "Stremio example" |     override var name = "Stremio example" | ||||||
|  | @ -53,7 +55,6 @@ class StremioProvider : MainAPI() { | ||||||
|         subtitleCallback: (SubtitleFile) -> Unit, |         subtitleCallback: (SubtitleFile) -> Unit, | ||||||
|         callback: (ExtractorLink) -> Unit |         callback: (ExtractorLink) -> Unit | ||||||
|     ): Boolean { |     ): Boolean { | ||||||
|         Log.i("Stremio", data) |  | ||||||
|         val res = tryParseJson<StreamsResponse>(app.get(data).text) ?: return false |         val res = tryParseJson<StreamsResponse>(app.get(data).text) ?: return false | ||||||
|         res.streams.forEach { stream -> |         res.streams.forEach { stream -> | ||||||
|             stream.runCallback(subtitleCallback, callback) |             stream.runCallback(subtitleCallback, callback) | ||||||
|  | @ -164,7 +165,9 @@ class StremioProvider : MainAPI() { | ||||||
|         val url: String?, |         val url: String?, | ||||||
|         val ytId: String?, |         val ytId: String?, | ||||||
|         val externalUrl: String?, |         val externalUrl: String?, | ||||||
|         val behaviorHints: JSONObject? |         val behaviorHints: JSONObject?, | ||||||
|  |         val infoHash: String?, | ||||||
|  |         val sources: List<String> = emptyList() | ||||||
|     ) { |     ) { | ||||||
|         suspend fun runCallback(subtitleCallback: (SubtitleFile) -> Unit, callback: (ExtractorLink) -> Unit) { |         suspend fun runCallback(subtitleCallback: (SubtitleFile) -> Unit, callback: (ExtractorLink) -> Unit) { | ||||||
|             if (url != null) { |             if (url != null) { | ||||||
|  | @ -205,6 +208,33 @@ class StremioProvider : MainAPI() { | ||||||
|             if (externalUrl != null) { |             if (externalUrl != null) { | ||||||
|                 loadExtractor(externalUrl, subtitleCallback, callback) |                 loadExtractor(externalUrl, subtitleCallback, callback) | ||||||
|             } |             } | ||||||
|  |             if (infoHash != null) { | ||||||
|  |                 val resp = app.get(TRACKER_LIST_URL).text | ||||||
|  |                 val otherTrackers = resp | ||||||
|  |                     .split("\n") | ||||||
|  |                     .filterIndexed{i, s -> i%2==0} | ||||||
|  |                     .filter{s -> !s.isNullOrEmpty()} | ||||||
|  |                     .map{it -> "&tr=$it"} | ||||||
|  |                     .joinToString("") | ||||||
|  |                  | ||||||
|  |                 val sourceTrackers = sources | ||||||
|  |                     .filter{it->it.startsWith("tracker:")} | ||||||
|  |                     .map{it->it.removePrefix("tracker:")} | ||||||
|  |                     .filter{s -> !s.isNullOrEmpty()} | ||||||
|  |                     .map{it -> "&tr=$it"} | ||||||
|  |                     .joinToString("") | ||||||
|  | 
 | ||||||
|  |                 val magnet = "magnet:?xt=urn:btih:${infoHash}${sourceTrackers}${otherTrackers}" | ||||||
|  |                 callback.invoke( | ||||||
|  |                     ExtractorLink( | ||||||
|  |                         name ?: "", | ||||||
|  |                         title ?: name ?: "", | ||||||
|  |                         magnet, | ||||||
|  |                         "", | ||||||
|  |                         Qualities.Unknown.value | ||||||
|  |                     ) | ||||||
|  |                 ) | ||||||
|  |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -1,5 +1,5 @@ | ||||||
| // use an integer for version numbers | // use an integer for version numbers | ||||||
| version = 4 | version = 7 | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| cloudstream { | cloudstream { | ||||||
|  |  | ||||||
|  | @ -17,11 +17,14 @@ import java.nio.charset.StandardCharsets | ||||||
| import java.security.MessageDigest | import java.security.MessageDigest | ||||||
| import java.security.NoSuchAlgorithmException | import java.security.NoSuchAlgorithmException | ||||||
| import javax.crypto.Cipher | import javax.crypto.Cipher | ||||||
|  | import javax.crypto.Cipher.DECRYPT_MODE | ||||||
|  | import javax.crypto.Cipher.ENCRYPT_MODE | ||||||
| import javax.crypto.spec.IvParameterSpec | import javax.crypto.spec.IvParameterSpec | ||||||
| import javax.crypto.spec.SecretKeySpec | import javax.crypto.spec.SecretKeySpec | ||||||
| import kotlin.math.roundToInt | import kotlin.math.roundToInt | ||||||
| 
 | 
 | ||||||
| class SuperStream : MainAPI() { | class SuperStream : MainAPI() { | ||||||
|  |     private val timeout = 120L | ||||||
|     override var name = "SuperStream" |     override var name = "SuperStream" | ||||||
|     override val hasMainPage = true |     override val hasMainPage = true | ||||||
|     override val hasChromecastSupport = true |     override val hasChromecastSupport = true | ||||||
|  | @ -78,7 +81,7 @@ class SuperStream : MainAPI() { | ||||||
|                     length++ |                     length++ | ||||||
|                 } |                 } | ||||||
|                 cipher.init( |                 cipher.init( | ||||||
|                     1, |                     ENCRYPT_MODE, | ||||||
|                     SecretKeySpec(bArr, ALGORITHM), |                     SecretKeySpec(bArr, ALGORITHM), | ||||||
|                     IvParameterSpec(iv.toByteArray()) |                     IvParameterSpec(iv.toByteArray()) | ||||||
|                 ) |                 ) | ||||||
|  | @ -90,6 +93,31 @@ class SuperStream : MainAPI() { | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  |         // Useful for deobfuscation | ||||||
|  |         fun decrypt(str: String, key: String, iv: String): String? { | ||||||
|  |             return try { | ||||||
|  |                 val cipher: Cipher = Cipher.getInstance(TRANSFORMATION) | ||||||
|  |                 val bArr = ByteArray(24) | ||||||
|  |                 val bytes: ByteArray = key.toByteArray() | ||||||
|  |                 var length = if (bytes.size <= 24) bytes.size else 24 | ||||||
|  |                 System.arraycopy(bytes, 0, bArr, 0, length) | ||||||
|  |                 while (length < 24) { | ||||||
|  |                     bArr[length] = 0 | ||||||
|  |                     length++ | ||||||
|  |                 } | ||||||
|  |                 cipher.init( | ||||||
|  |                     DECRYPT_MODE, | ||||||
|  |                     SecretKeySpec(bArr, ALGORITHM), | ||||||
|  |                     IvParameterSpec(iv.toByteArray()) | ||||||
|  |                 ) | ||||||
|  |                 val inputStr = Base64.decode(str.toByteArray(), Base64.DEFAULT) | ||||||
|  |                 cipher.doFinal(inputStr).decodeToString() | ||||||
|  |             } catch (e: Exception) { | ||||||
|  |                 e.printStackTrace() | ||||||
|  |                 null | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|         fun md5(str: String): String? { |         fun md5(str: String): String? { | ||||||
|             return MD5Util.md5(str)?.let { HexDump.toHexString(it).lowercase() } |             return MD5Util.md5(str)?.let { HexDump.toHexString(it).lowercase() } | ||||||
|         } |         } | ||||||
|  | @ -155,7 +183,7 @@ class SuperStream : MainAPI() { | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     private suspend fun queryApi(query: String): NiceResponse { |     private suspend fun queryApi(query: String, useAlternativeApi: Boolean): NiceResponse { | ||||||
|         val encryptedQuery = CipherUtils.encrypt(query, key, iv)!! |         val encryptedQuery = CipherUtils.encrypt(query, key, iv)!! | ||||||
|         val appKeyHash = CipherUtils.md5(appKey)!! |         val appKeyHash = CipherUtils.md5(appKey)!! | ||||||
|         val newBody = |         val newBody = | ||||||
|  | @ -172,16 +200,20 @@ class SuperStream : MainAPI() { | ||||||
|             "data" to base64Body, |             "data" to base64Body, | ||||||
|             "appid" to "27", |             "appid" to "27", | ||||||
|             "platform" to "android", |             "platform" to "android", | ||||||
|             "version" to "129", |             "version" to appVersionCode, | ||||||
|             // Probably best to randomize this |             // Probably best to randomize this | ||||||
|             "medium" to "Website&token$token" |             "medium" to "Website&token$token" | ||||||
|         ) |         ) | ||||||
| 
 | 
 | ||||||
|         return app.post(apiUrl, headers = headers, data = data) |         val url = if (useAlternativeApi) secondApiUrl else apiUrl | ||||||
|  |         return app.post(url, headers = headers, data = data, timeout = timeout) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     private suspend inline fun <reified T : Any> queryApiParsed(query: String): T { |     private suspend inline fun <reified T : Any> queryApiParsed( | ||||||
|         return queryApi(query).parsed() |         query: String, | ||||||
|  |         useAlternativeApi: Boolean = true | ||||||
|  |     ): T { | ||||||
|  |         return queryApi(query, useAlternativeApi).parsed() | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     private fun getExpiryDate(): Long { |     private fun getExpiryDate(): Long { | ||||||
|  | @ -219,21 +251,31 @@ class SuperStream : MainAPI() { | ||||||
|     // Free Tibet, The Tienanmen Square protests of 1989 |     // Free Tibet, The Tienanmen Square protests of 1989 | ||||||
|     private val iv = base64Decode("d0VpcGhUbiE=") |     private val iv = base64Decode("d0VpcGhUbiE=") | ||||||
|     private val key = base64Decode("MTIzZDZjZWRmNjI2ZHk1NDIzM2FhMXc2") |     private val key = base64Decode("MTIzZDZjZWRmNjI2ZHk1NDIzM2FhMXc2") | ||||||
|  | 
 | ||||||
|     private val ip = base64Decode("aHR0cHM6Ly8xNTIuMzIuMTQ5LjE2MA==") |     private val ip = base64Decode("aHR0cHM6Ly8xNTIuMzIuMTQ5LjE2MA==") | ||||||
|     private val apiUrl = |     private val apiUrl = | ||||||
|         "$ip${base64Decode("L2FwaS9hcGlfY2xpZW50L2luZGV4Lw==")}" |         "$ip${base64Decode("L2FwaS9hcGlfY2xpZW50L2luZGV4Lw==")}" | ||||||
|  | 
 | ||||||
|  |     // Another url because the first one sucks at searching | ||||||
|  |     // This one was revealed to me in a dream | ||||||
|  |     private val secondApiUrl = | ||||||
|  |         base64Decode("aHR0cHM6Ly9tYnBhcGkuc2hlZ3UubmV0L2FwaS9hcGlfY2xpZW50L2luZGV4Lw==") | ||||||
|  | 
 | ||||||
|     private val appKey = base64Decode("bW92aWVib3g=") |     private val appKey = base64Decode("bW92aWVib3g=") | ||||||
|     private val appId = base64Decode("Y29tLnRkby5zaG93Ym94") |     private val appId = base64Decode("Y29tLnRkby5zaG93Ym94") | ||||||
|  |     private val appIdSecond = base64Decode("Y29tLm1vdmllYm94cHJvLmFuZHJvaWQ=") | ||||||
|  |     private val appVersion = "14.7" | ||||||
|  |     private val appVersionCode = "160" | ||||||
| 
 | 
 | ||||||
|     override suspend fun getMainPage(page: Int, request: MainPageRequest): HomePageResponse { |     override suspend fun getMainPage(page: Int, request: MainPageRequest): HomePageResponse { | ||||||
|         val hideNsfw = if (settingsForProvider.enableAdult) 0 else 1 |         val hideNsfw = if (settingsForProvider.enableAdult) 0 else 1 | ||||||
|         val json = queryApi( |         val data = queryApiParsed<DataJSON>( | ||||||
|             """{"childmode":"$hideNsfw","app_version":"11.5","appid":"$appId","module":"Home_list_type_v2","channel":"Website","page":"$page","lang":"en","type":"all","pagelimit":"10","expired_date":"${getExpiryDate()}","platform":"android"} |             """{"childmode":"$hideNsfw","app_version":"$appVersion","appid":"$appIdSecond","module":"Home_list_type_v5","channel":"Website","page":"$page","lang":"en","type":"all","pagelimit":"10","expired_date":"${getExpiryDate()}","platform":"android"} | ||||||
|             """.trimIndent() |             """.trimIndent() | ||||||
|         ).text |         ) | ||||||
| 
 | 
 | ||||||
|         // Cut off the first row (featured) |         // Cut off the first row (featured) | ||||||
|         val pages = parseJson<DataJSON>(json).data.let { it.subList(minOf(it.size, 1), it.size) } |         val pages = data.data.let { it.subList(minOf(it.size, 1), it.size) } | ||||||
|             .mapNotNull { |             .mapNotNull { | ||||||
|                 var name = it.name |                 var name = it.name | ||||||
|                 if (name.isNullOrEmpty()) name = "Featured" |                 if (name.isNullOrEmpty()) name = "Featured" | ||||||
|  | @ -270,7 +312,10 @@ class SuperStream : MainAPI() { | ||||||
|         fun toSearchResponse(api: MainAPI): MovieSearchResponse? { |         fun toSearchResponse(api: MainAPI): MovieSearchResponse? { | ||||||
|             return api.newMovieSearchResponse( |             return api.newMovieSearchResponse( | ||||||
|                 this.title ?: "", |                 this.title ?: "", | ||||||
|                 LoadData(this.id ?: this.mid ?: return null, this.boxType ?: ResponseTypes.Movies.value).toJson(), |                 LoadData( | ||||||
|  |                     this.id ?: this.mid ?: return null, | ||||||
|  |                     this.boxType ?: ResponseTypes.Movies.value | ||||||
|  |                 ).toJson(), | ||||||
|                 ResponseTypes.getResponseType(this.boxType).toTvType(), |                 ResponseTypes.getResponseType(this.boxType).toTvType(), | ||||||
|                 false |                 false | ||||||
|             ) { |             ) { | ||||||
|  | @ -289,8 +334,8 @@ class SuperStream : MainAPI() { | ||||||
|         val hideNsfw = if (settingsForProvider.enableAdult) 0 else 1 |         val hideNsfw = if (settingsForProvider.enableAdult) 0 else 1 | ||||||
|         val apiQuery = |         val apiQuery = | ||||||
|             // Originally 8 pagelimit |             // Originally 8 pagelimit | ||||||
|             """{"childmode":"$hideNsfw","app_version":"11.5","appid":"$appId","module":"Search3","channel":"Website","page":"1","lang":"en","type":"all","keyword":"$query","pagelimit":"20","expired_date":"${getExpiryDate()}","platform":"android"}""" |             """{"childmode":"$hideNsfw","app_version":"$appVersion","appid":"$appIdSecond","module":"Search3","channel":"Website","page":"1","lang":"en","type":"all","keyword":"$query","pagelimit":"20","expired_date":"${getExpiryDate()}","platform":"android"}""" | ||||||
|         val searchResponse = parseJson<MainData>(queryApi(apiQuery).text).data.mapNotNull { |         val searchResponse = queryApiParsed<MainData>(apiQuery, true).data.mapNotNull { | ||||||
|             it.toSearchResponse(this) |             it.toSearchResponse(this) | ||||||
|         } |         } | ||||||
|         return searchResponse |         return searchResponse | ||||||
|  | @ -471,7 +516,7 @@ class SuperStream : MainAPI() { | ||||||
|         val hideNsfw = if (settingsForProvider.enableAdult) 0 else 1 |         val hideNsfw = if (settingsForProvider.enableAdult) 0 else 1 | ||||||
|         if (isMovie) { // 1 = Movie |         if (isMovie) { // 1 = Movie | ||||||
|             val apiQuery = |             val apiQuery = | ||||||
|                 """{"childmode":"$hideNsfw","uid":"","app_version":"11.5","appid":"$appId","module":"Movie_detail","channel":"Website","mid":"${loadData.id}","lang":"en","expired_date":"${getExpiryDate()}","platform":"android","oss":"","group":""}""" |                 """{"childmode":"$hideNsfw","uid":"","app_version":"$appVersion","appid":"$appIdSecond","module":"Movie_detail","channel":"Website","mid":"${loadData.id}","lang":"en","expired_date":"${getExpiryDate()}","platform":"android","oss":"","group":""}""" | ||||||
|             val data = (queryApiParsed<MovieDataProp>(apiQuery)).data |             val data = (queryApiParsed<MovieDataProp>(apiQuery)).data | ||||||
|                 ?: throw RuntimeException("API error") |                 ?: throw RuntimeException("API error") | ||||||
| 
 | 
 | ||||||
|  | @ -498,13 +543,13 @@ class SuperStream : MainAPI() { | ||||||
|             } |             } | ||||||
|         } else { // 2 Series |         } else { // 2 Series | ||||||
|             val apiQuery = |             val apiQuery = | ||||||
|                 """{"childmode":"$hideNsfw","uid":"","app_version":"11.5","appid":"$appId","module":"TV_detail_1","display_all":"1","channel":"Website","lang":"en","expired_date":"${getExpiryDate()}","platform":"android","tid":"${loadData.id}"}""" |                 """{"childmode":"$hideNsfw","uid":"","app_version":"$appVersion","appid":"$appIdSecond","module":"TV_detail_1","display_all":"1","channel":"Website","lang":"en","expired_date":"${getExpiryDate()}","platform":"android","tid":"${loadData.id}"}""" | ||||||
|             val data = (queryApiParsed<SeriesDataProp>(apiQuery)).data |             val data = (queryApiParsed<SeriesDataProp>(apiQuery)).data | ||||||
|                 ?: throw RuntimeException("API error") |                 ?: throw RuntimeException("API error") | ||||||
| 
 | 
 | ||||||
|             val episodes = data.season.mapNotNull { |             val episodes = data.season.mapNotNull { | ||||||
|                 val seasonQuery = |                 val seasonQuery = | ||||||
|                     """{"childmode":"$hideNsfw","app_version":"11.5","year":"0","appid":"$appId","module":"TV_episode","display_all":"1","channel":"Website","season":"$it","lang":"en","expired_date":"${getExpiryDate()}","platform":"android","tid":"${loadData.id}"}""" |                     """{"childmode":"$hideNsfw","app_version":"$appVersion","year":"0","appid":"$appIdSecond","module":"TV_episode","display_all":"1","channel":"Website","season":"$it","lang":"en","expired_date":"${getExpiryDate()}","platform":"android","tid":"${loadData.id}"}""" | ||||||
|                 (queryApiParsed<SeriesSeasonProp>(seasonQuery)).data |                 (queryApiParsed<SeriesSeasonProp>(seasonQuery)).data | ||||||
|             }.flatten() |             }.flatten() | ||||||
| 
 | 
 | ||||||
|  | @ -646,6 +691,7 @@ class SuperStream : MainAPI() { | ||||||
|         val parsed = parseJson<LinkData>(data) |         val parsed = parseJson<LinkData>(data) | ||||||
| 
 | 
 | ||||||
|         // No childmode when getting links |         // No childmode when getting links | ||||||
|  |         // New api does not return video links :( | ||||||
|         val query = if (parsed.type == ResponseTypes.Movies.value) { |         val query = if (parsed.type == ResponseTypes.Movies.value) { | ||||||
|             """{"childmode":"0","uid":"","app_version":"11.5","appid":"$appId","module":"Movie_downloadurl_v3","channel":"Website","mid":"${parsed.id}","lang":"","expired_date":"${getExpiryDate()}","platform":"android","oss":"1","group":""}""" |             """{"childmode":"0","uid":"","app_version":"11.5","appid":"$appId","module":"Movie_downloadurl_v3","channel":"Website","mid":"${parsed.id}","lang":"","expired_date":"${getExpiryDate()}","platform":"android","oss":"1","group":""}""" | ||||||
|         } else { |         } else { | ||||||
|  | @ -654,7 +700,7 @@ class SuperStream : MainAPI() { | ||||||
|             """{"childmode":"0","app_version":"11.5","module":"TV_downloadurl_v3","channel":"Website","episode":"$episode","expired_date":"${getExpiryDate()}","platform":"android","tid":"${parsed.id}","oss":"1","uid":"","appid":"$appId","season":"$season","lang":"en","group":""}""" |             """{"childmode":"0","app_version":"11.5","module":"TV_downloadurl_v3","channel":"Website","episode":"$episode","expired_date":"${getExpiryDate()}","platform":"android","tid":"${parsed.id}","oss":"1","uid":"","appid":"$appId","season":"$season","lang":"en","group":""}""" | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         val linkData = queryApiParsed<LinkDataProp>(query) |         val linkData = queryApiParsed<LinkDataProp>(query, false) | ||||||
|         linkData.data?.list?.forEach { |         linkData.data?.list?.forEach { | ||||||
|             callback.invoke(it.toExtractorLink() ?: return@forEach) |             callback.invoke(it.toExtractorLink() ?: return@forEach) | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  | @ -0,0 +1,36 @@ | ||||||
|  | package com.lagradost | ||||||
|  | 
 | ||||||
|  | import com.lagradost.cloudstream3.* | ||||||
|  | import com.lagradost.cloudstream3.utils.AppUtils.tryParseJson | ||||||
|  | import com.lagradost.cloudstream3.utils.AppUtils.toJson | ||||||
|  | import kotlinx.coroutines.delay | ||||||
|  | import org.json.JSONArray | ||||||
|  | import org.json.JSONObject | ||||||
|  | import android.util.Log | ||||||
|  | 
 | ||||||
|  | public object CaptchaSolver { | ||||||
|  |     suspend fun predictFace(url: String): String? { | ||||||
|  |         val img = "data:image/jpeg;base64," + base64Encode(app.get(url).body.bytes()) | ||||||
|  |         val reqData = HFRequest(listOf(img)).toJson() | ||||||
|  |         val res = app.post("https://yuqi-gender-classifier.hf.space/api/queue/push/", json = reqData).text | ||||||
|  |         val request = tryParseJson<JSONObject>(res) | ||||||
|  |         for (i in 1..5) { | ||||||
|  |             delay(500L) | ||||||
|  |             val document = app.post("https://yuqi-gender-classifier.hf.space/api/queue/status/", json=request?.toJson()).text | ||||||
|  |             val status = tryParseJson<JSONObject>(document) | ||||||
|  |             if (status?.get("status") != "COMPLETE") continue | ||||||
|  |             return (((status.get("data") as? JSONObject?) | ||||||
|  |                 ?.get("data") as? JSONArray?) | ||||||
|  |                 ?.get(0) as? JSONObject?) | ||||||
|  |                 ?.get("label") as String? | ||||||
|  |         } | ||||||
|  |         return null | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private data class HFRequest( | ||||||
|  |         val data: List<String>, | ||||||
|  |         val action: String = "predict", | ||||||
|  |         val fn_index: Int = 0, | ||||||
|  |         val session_hash: String = "aaaaaaaaaaa" | ||||||
|  |     ) | ||||||
|  | } | ||||||
|  | @ -6,9 +6,8 @@ import com.lagradost.cloudstream3.metaproviders.TmdbProvider | ||||||
| import com.lagradost.cloudstream3.utils.AppUtils.tryParseJson | import com.lagradost.cloudstream3.utils.AppUtils.tryParseJson | ||||||
| import com.lagradost.cloudstream3.utils.ExtractorLink | import com.lagradost.cloudstream3.utils.ExtractorLink | ||||||
| import com.lagradost.cloudstream3.utils.loadExtractor | import com.lagradost.cloudstream3.utils.loadExtractor | ||||||
| import kotlinx.coroutines.delay | import okhttp3.FormBody | ||||||
| import org.json.JSONArray | import android.util.Log | ||||||
| import org.json.JSONObject |  | ||||||
| 
 | 
 | ||||||
| class SuperembedProvider : TmdbProvider() { | class SuperembedProvider : TmdbProvider() { | ||||||
|     override var mainUrl = "https://seapi.link" |     override var mainUrl = "https://seapi.link" | ||||||
|  | @ -54,40 +53,31 @@ class SuperembedProvider : TmdbProvider() { | ||||||
|         val url: String |         val url: String | ||||||
|     ) { |     ) { | ||||||
|         suspend fun getIframeContents(): String? { |         suspend fun getIframeContents(): String? { | ||||||
|             val document = app.get(url) |             var document = app.get(url) | ||||||
|  |             for (i in 1..5) { | ||||||
|  |                 if ("captcha-message" in document.text) { | ||||||
|  |                     val soup = document.document | ||||||
|  |                     val prompt = soup.selectFirst(".captcha-message")?.text() ?: continue | ||||||
|  |                     val captchaId = soup.selectFirst("input[name=\"captcha_id\"]")?.attr("value") ?: continue | ||||||
|  |                     val promptGender = if ("female" in prompt) "female" else "male" | ||||||
|  |                     val checkboxes = soup.select(".captcha-checkbox").mapNotNull { it ->  | ||||||
|  |                         val img = it.selectFirst("img")?.attr("src") ?: return@mapNotNull null | ||||||
|  |                         val gender = CaptchaSolver.predictFace("https://streamembed.net${img}") ?: return@mapNotNull null | ||||||
|  |                         if (gender != promptGender) return@mapNotNull null | ||||||
|  |                         return@mapNotNull it.selectFirst("input")?.attr("value") | ||||||
|  |                     } | ||||||
|  |                     val formData = FormBody.Builder().apply { | ||||||
|  |                         add("captcha_id", captchaId) | ||||||
|  |                         checkboxes.forEach { check -> | ||||||
|  |                             add("captcha_answer[]", check) | ||||||
|  |                         } | ||||||
|  |                     }.build() | ||||||
|  |                     document = app.post(url, requestBody=formData) | ||||||
|  |                 } else { break } | ||||||
|  |             } | ||||||
|             val regex = "<iframe[^+]+\\+(?:window\\.)?atob\\(['\"]([-A-Za-z0-9+/=]+)".toRegex() |             val regex = "<iframe[^+]+\\+(?:window\\.)?atob\\(['\"]([-A-Za-z0-9+/=]+)".toRegex() | ||||||
|             val encoded = regex.find(document.text)?.groupValues?.get(1) ?: return null |             val encoded = regex.find(document.text)?.groupValues?.get(1) ?: return null | ||||||
|             return base64Decode(encoded) |             return base64Decode(encoded) | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 |  | ||||||
|     /* |  | ||||||
|     private object CaptchaSolver { |  | ||||||
|         private enum class Gender { Female, Male } |  | ||||||
|         private suspend fun predictFace(url: String): Gender? { |  | ||||||
|             val img = "data:image/jpeg;base64," + base64Encode(app.get(url).body.bytes()) |  | ||||||
|             val res = app.post("https://hf.space/embed/njgroene/age-gender-profilepic/api/queue/push/ HTTP/1.1", json = HFRequest( |  | ||||||
|                 listOf(img))).text |  | ||||||
|             val request = tryParseJson<JSONObject>(res) |  | ||||||
|             for (i in 1..5) { |  | ||||||
|                 delay(500L) |  | ||||||
|                 val document = app.post("https://hf.space/embed/njgroene/age-gender-profilepic/api/queue/status/", json=request).text |  | ||||||
|                 val status = tryParseJson<JSONObject>(document) |  | ||||||
|                 if (status?.get("status") != "COMPLETE") continue |  | ||||||
|                 val pred = (((status.get("data") as? JSONObject?) |  | ||||||
|                     ?.get("data") as? JSONArray?) |  | ||||||
|                     ?.get(0) as? String?) ?: return null |  | ||||||
|                 return if ("Male" in pred) Gender.Male |  | ||||||
|                 else if ("Female" in pred) Gender.Female |  | ||||||
|                 else null |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         private data class HFRequest( |  | ||||||
|             val data: List<String>, |  | ||||||
|             val action: String = "predict", |  | ||||||
|             val fn_index: Int = 0, |  | ||||||
|             val session_hash: String = "aaaaaaaaaaa" |  | ||||||
|         ) |  | ||||||
|     }*/ |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -80,7 +80,7 @@ subprojects { | ||||||
|         implementation("com.github.Blatzar:NiceHttp:0.3.2") // http library |         implementation("com.github.Blatzar:NiceHttp:0.3.2") // http library | ||||||
|         implementation("com.fasterxml.jackson.module:jackson-module-kotlin:2.13.1") |         implementation("com.fasterxml.jackson.module:jackson-module-kotlin:2.13.1") | ||||||
|         implementation("org.jsoup:jsoup:1.13.1") // html parser |         implementation("org.jsoup:jsoup:1.13.1") // html parser | ||||||
|         implementation("org.jetbrains.kotlinx:kotlinx-coroutines-android:1.6.4") // html parser |         implementation("org.jetbrains.kotlinx:kotlinx-coroutines-android:1.6.4") // delay() | ||||||
| 
 | 
 | ||||||
|         //run JS |         //run JS | ||||||
|         implementation("org.mozilla:rhino:1.7.14") |         implementation("org.mozilla:rhino:1.7.14") | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue