Provider Soap2Day (#757)

* Soap2Day

* small changes
This commit is contained in:
Stormunblessed 2022-03-07 14:18:51 +00:00 committed by GitHub
parent da053139e5
commit 7e3e13c224
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
2 changed files with 219 additions and 0 deletions

View file

@ -89,6 +89,7 @@ object APIHolder {
AnimePaheProvider(), AnimePaheProvider(),
NineAnimeProvider(), NineAnimeProvider(),
AnimeWorldProvider(), AnimeWorldProvider(),
SoaptwoDayProvider(),
CrossTmdbProvider(), CrossTmdbProvider(),
) )

View file

@ -0,0 +1,218 @@
package com.lagradost.cloudstream3.movieproviders
import com.fasterxml.jackson.annotation.JsonProperty
import com.lagradost.cloudstream3.*
import com.lagradost.cloudstream3.mvvm.logError
import com.lagradost.cloudstream3.utils.*
import com.lagradost.cloudstream3.utils.AppUtils.parseJson
import java.util.*
class SoaptwoDayProvider:MainAPI() {
override val mainUrl = "https://secretlink.xyz" //Probably a rip off, but it has no captcha
override val name = "Soap2Day"
override val hasMainPage = true
override val hasChromecastSupport = true
override val hasDownloadSupport = true
override val supportedTypes = setOf(
TvType.Movie,
TvType.TvSeries,
)
override suspend fun getMainPage(): HomePageResponse {
val items = ArrayList<HomePageList>()
val urls = listOf(
Pair("$mainUrl/movielist/", "Movies"),
Pair("$mainUrl/tvlist/", "TV Series"),
)
for ((url, name) in urls) {
try {
val soup = app.get(url).document
val home = soup.select("div.container div.row div.col-sm-12.col-lg-12 div.row div.col-sm-12.col-lg-12 .col-xs-6").map {
val title = it.selectFirst("h5 a").text()
val link = it.selectFirst("a").attr("href")
TvSeriesSearchResponse(
title,
link,
this.name,
TvType.TvSeries,
fixUrl(it.selectFirst("img").attr("src")),
null,
null,
)
}
items.add(HomePageList(name, home))
} catch (e: Exception) {
logError(e)
}
}
return HomePageResponse(items)
}
override suspend fun search(query: String): List<SearchResponse> {
val doc = app.get("$mainUrl/search/keyword/$query").document
return doc.select("div.container div.row div.col-sm-12.col-lg-12 div.row div.col-sm-12.col-lg-12 .col-xs-6").map {
val title = it.selectFirst("h5 a").text()
val image = fixUrl(it.selectFirst("img").attr("src"))
val href = fixUrl(it.selectFirst("a").attr("href"))
TvSeriesSearchResponse(
title,
href,
this.name,
TvType.TvSeries,
image,
null,
null
)
}
}
override suspend fun load(url: String): LoadResponse? {
val soup = app.get(url, timeout = 120).document
val title = soup.selectFirst(".hidden-lg > div:nth-child(1) > h4").text()
val description = soup.selectFirst("p#wrap")?.text()?.trim()
val poster = soup.selectFirst(".col-md-5 > div:nth-child(1) > div:nth-child(1) > img").attr("src")
val episodes = soup.select("div.alert > div > div > a").map {
val link = it.attr("href")
val name = it.text().replace(Regex("(^(\\d+)\\.)"),"")
TvSeriesEpisode(
name,
null,
null,
fixUrl(link)
)
}
val tvType = if (episodes.isEmpty()) TvType.Movie else TvType.TvSeries
return when (tvType) {
TvType.TvSeries -> {
TvSeriesLoadResponse(
title,
url,
this.name,
tvType,
episodes.reversed(),
fixUrl(poster),
null,
description,
)
}
TvType.Movie -> {
MovieLoadResponse(
title,
url,
this.name,
tvType,
url,
fixUrl(poster),
null,
description,
)
}
else -> null
}
}
data class ServerJson (
@JsonProperty("0") val zero: String?,
@JsonProperty("key") val key: Boolean?,
@JsonProperty("val") val stream: String?,
@JsonProperty("val_bak") val streambackup: String?,
@JsonProperty("pos") val pos: Int?,
@JsonProperty("type") val type: String?,
@JsonProperty("subs") val subs: List<Subs>?,
@JsonProperty("prev_epi_title") val prevEpiTitle: String?,
@JsonProperty("prev_epi_url") val prevEpiUrl: String?,
@JsonProperty("next_epi_title") val nextEpiTitle: String?,
@JsonProperty("next_epi_url") val nextEpiUrl: String?
)
data class Subs (
@JsonProperty("id") val id: Int?,
@JsonProperty("movieId") val movieId: Int?,
@JsonProperty("tvId") val tvId: Int?,
@JsonProperty("episodeId") val episodeId: Int?,
@JsonProperty("default") val default: Int?,
@JsonProperty("IsShow") val IsShow: Int?,
@JsonProperty("name") val name: String,
@JsonProperty("path") val path: String?,
@JsonProperty("downlink") val downlink: String?,
@JsonProperty("source_file_name") val sourceFileName: String?,
@JsonProperty("createtime") val createtime: Int?
)
override suspend fun loadLinks(
data: String,
isCasting: Boolean,
subtitleCallback: (SubtitleFile) -> Unit,
callback: (ExtractorLink) -> Unit
): Boolean {
val doc = app.get(data).document
val idplayer = doc.selectFirst("#divU")?.text()
val idplayer2 = doc.selectFirst("#divP")?.text()
val movieid = doc.selectFirst("div.row input#hId").attr("value")
val tvType = try {
doc.selectFirst(".col-md-5 > div:nth-child(1) > div:nth-child(1) > img").attr("src") ?: ""
} catch (e: Exception) {
""
}
val ajaxlink = if (tvType.contains("movie")) "$mainUrl/home/index/GetMInfoAjax" else "$mainUrl/home/index/GetEInfoAjax"
listOf(
idplayer,
idplayer2,
).mapNotNull { playerID ->
val url = app.post(ajaxlink,
headers = mapOf(
"Host" to "secretlink.xyz",
"User-Agent" to USER_AGENT,
"Accept" to "application/json, text/javascript, */*; q=0.01",
"Accept-Language" to "en-US,en;q=0.5",
"Content-Type" to "application/x-www-form-urlencoded; charset=UTF-8",
"X-Requested-With" to "XMLHttpRequest",
"Origin" to "https://secretlink.xyz",
"DNT" to "1",
"Connection" to "keep-alive",
"Referer" to data,
"Sec-Fetch-Dest" to "empty",
"Sec-Fetch-Mode" to "cors",
"Sec-Fetch-Site" to "same-origin",),
data = mapOf(
Pair("pass",movieid),
Pair("param",playerID),
)
).text.replace("\\\"","\"").replace("\"{","{").replace("}\"","}")
.replace("\\\\\\/","\\/")
val json = parseJson<ServerJson>(url)
listOf(
json.stream,
json.streambackup
).filterNotNull().apmap { stream ->
val cleanstreamurl = stream.replace("\\/","/").replace("\\\\\\","")
if (cleanstreamurl.isNotBlank()) {
callback(ExtractorLink(
"Soap2Day",
"Soap2Day",
cleanstreamurl,
"https://soap2day.ac",
Qualities.Unknown.value,
isM3u8 = false
))
}
}
json.subs?.forEach { subtitle ->
val sublink = mainUrl+subtitle.path
listOf(
sublink,
subtitle.downlink
).mapNotNull { subs ->
if (subs != null) {
if (subs.isNotBlank()) {
subtitleCallback(
SubtitleFile(subtitle.name, subs)
)
}
}
}
}
}
return true
}
}