Merge branch 'master' of https://github.com/antonydp/cloudstream-extensions-multilingual
This commit is contained in:
commit
e7609e8da5
|
@ -1,5 +1,5 @@
|
||||||
// use an integer for version numbers
|
// use an integer for version numbers
|
||||||
version = 1
|
version = 2
|
||||||
|
|
||||||
|
|
||||||
cloudstream {
|
cloudstream {
|
||||||
|
|
|
@ -2,7 +2,7 @@ package com.lagradost
|
||||||
|
|
||||||
import com.lagradost.cloudstream3.*
|
import com.lagradost.cloudstream3.*
|
||||||
import com.lagradost.cloudstream3.utils.*
|
import com.lagradost.cloudstream3.utils.*
|
||||||
|
import org.jsoup.nodes.Document
|
||||||
|
|
||||||
class CalcioStreamingProvider : MainAPI() {
|
class CalcioStreamingProvider : MainAPI() {
|
||||||
override var lang = "it"
|
override var lang = "it"
|
||||||
|
@ -58,12 +58,21 @@ class CalcioStreamingProvider : MainAPI() {
|
||||||
poster,
|
poster,
|
||||||
plot = Matchstart
|
plot = Matchstart
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private fun matchFound(document: Document) : Boolean {
|
||||||
|
return Regex(""""((.|\n)*?).";""").containsMatchIn(
|
||||||
|
getAndUnpack(
|
||||||
|
document.toString()
|
||||||
|
))
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun getUrl(document: Document):String{
|
||||||
|
return Regex(""""((.|\n)*?).";""").find(
|
||||||
|
getAndUnpack(
|
||||||
|
document.toString()
|
||||||
|
))!!.value.replace("""src="""", "").replace(""""""", "").replace(";", "")
|
||||||
|
}
|
||||||
|
|
||||||
private suspend fun extractVideoLinks(
|
private suspend fun extractVideoLinks(
|
||||||
url: String,
|
url: String,
|
||||||
|
@ -71,25 +80,28 @@ class CalcioStreamingProvider : MainAPI() {
|
||||||
) {
|
) {
|
||||||
val document = app.get(url).document
|
val document = app.get(url).document
|
||||||
document.select("button.btn").forEach { button ->
|
document.select("button.btn").forEach { button ->
|
||||||
val link1 = button.attr("data-link")
|
var link = button.attr("data-link")
|
||||||
val doc2 = app.get(link1).document
|
var oldLink = link
|
||||||
val truelink = doc2.selectFirst("iframe")!!.attr("src")
|
var videoNotFound = true
|
||||||
val newpage = app.get(truelink, referer = link1).document
|
while (videoNotFound) {
|
||||||
val streamurl = Regex(""""((.|\n)*?).";""").find(
|
val doc = app.get(link).document
|
||||||
getAndUnpack(
|
link = doc.selectFirst("iframe")?.attr("src") ?: break
|
||||||
newpage.select("script")[6].childNode(0).toString()
|
val newpage = app.get(fixUrl(link), referer = oldLink).document
|
||||||
))!!.value.replace("""src="""", "").replace(""""""", "").replace(";", "")
|
oldLink = link
|
||||||
|
if (newpage.select("script").size >= 6 && matchFound(newpage)){
|
||||||
callback(
|
videoNotFound = false
|
||||||
ExtractorLink(
|
callback(
|
||||||
this.name,
|
ExtractorLink(
|
||||||
button.text(),
|
this.name,
|
||||||
streamurl,
|
button.text(),
|
||||||
truelink,
|
getUrl(newpage),
|
||||||
quality = 0,
|
fixUrl(link),
|
||||||
true
|
quality = 0,
|
||||||
)
|
true
|
||||||
)
|
)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
// use an integer for version numbers
|
// use an integer for version numbers
|
||||||
version = 2
|
version = 3
|
||||||
|
|
||||||
|
|
||||||
cloudstream {
|
cloudstream {
|
||||||
|
|
|
@ -10,7 +10,7 @@ import org.jsoup.nodes.Element
|
||||||
|
|
||||||
class CineBlog01Provider : MainAPI() {
|
class CineBlog01Provider : MainAPI() {
|
||||||
override var lang = "it"
|
override var lang = "it"
|
||||||
override var mainUrl = "https://www.cineblog01.moe"
|
override var mainUrl = "https://www.cineblog01.mom"
|
||||||
override var name = "CineBlog01"
|
override var name = "CineBlog01"
|
||||||
override val hasMainPage = true
|
override val hasMainPage = true
|
||||||
override val hasChromecastSupport = true
|
override val hasChromecastSupport = true
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
// use an integer for version numbers
|
// use an integer for version numbers
|
||||||
version = 2
|
version = 3
|
||||||
|
|
||||||
cloudstream {
|
cloudstream {
|
||||||
language = "it"
|
language = "it"
|
||||||
|
|
|
@ -11,7 +11,7 @@ import org.jsoup.nodes.Element
|
||||||
|
|
||||||
class EurostreamingProvider : MainAPI() {
|
class EurostreamingProvider : MainAPI() {
|
||||||
override var lang = "it"
|
override var lang = "it"
|
||||||
override var mainUrl = "https://eurostreaming.taxi"
|
override var mainUrl = "https://eurostreaming.expert"
|
||||||
override var name = "Eurostreaming"
|
override var name = "Eurostreaming"
|
||||||
override val hasMainPage = true
|
override val hasMainPage = true
|
||||||
override val hasChromecastSupport = true
|
override val hasChromecastSupport = true
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
// use an integer for version numbers
|
// use an integer for version numbers
|
||||||
version = 4
|
version = 5
|
||||||
|
|
||||||
|
|
||||||
cloudstream {
|
cloudstream {
|
||||||
|
|
|
@ -16,7 +16,7 @@ import com.lagradost.cloudstream3.network.CloudflareKiller
|
||||||
|
|
||||||
class FilmpertuttiProvider : MainAPI() {
|
class FilmpertuttiProvider : MainAPI() {
|
||||||
override var lang = "it"
|
override var lang = "it"
|
||||||
override var mainUrl = "https://filmpertutti.skin"
|
override var mainUrl = "https://filmpertutti.tips"
|
||||||
override var name = "FilmPerTutti"
|
override var name = "FilmPerTutti"
|
||||||
override val hasMainPage = true
|
override val hasMainPage = true
|
||||||
override val hasChromecastSupport = true
|
override val hasChromecastSupport = true
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
// use an integer for version numbers
|
// use an integer for version numbers
|
||||||
version = 1
|
version = 2
|
||||||
|
|
||||||
|
|
||||||
cloudstream {
|
cloudstream {
|
||||||
|
|
|
@ -2,9 +2,11 @@ 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.network.CloudflareKiller
|
||||||
import com.lagradost.cloudstream3.utils.*
|
import com.lagradost.cloudstream3.utils.*
|
||||||
import com.lagradost.cloudstream3.utils.AppUtils.toJson
|
import com.lagradost.cloudstream3.utils.AppUtils.toJson
|
||||||
import com.lagradost.cloudstream3.utils.AppUtils.tryParseJson
|
import com.lagradost.cloudstream3.utils.AppUtils.tryParseJson
|
||||||
|
import org.jsoup.nodes.Element
|
||||||
|
|
||||||
class StarLiveProvider : MainAPI() {
|
class StarLiveProvider : MainAPI() {
|
||||||
override var mainUrl = "https://starlive.xyz"
|
override var mainUrl = "https://starlive.xyz"
|
||||||
|
@ -13,72 +15,51 @@ class StarLiveProvider : MainAPI() {
|
||||||
override var lang = "it"
|
override var lang = "it"
|
||||||
override val hasChromecastSupport = true
|
override val hasChromecastSupport = true
|
||||||
override val supportedTypes = setOf(TvType.Live)
|
override val supportedTypes = setOf(TvType.Live)
|
||||||
|
private val interceptor = CloudflareKiller()
|
||||||
private data class LinkParser(
|
|
||||||
@JsonProperty("link") val link: String,
|
|
||||||
@JsonProperty("lang") val language: String,
|
|
||||||
@JsonProperty("name") val name: String
|
|
||||||
)
|
|
||||||
|
|
||||||
private data class MatchDataParser(
|
|
||||||
@JsonProperty("time") val time: String,
|
|
||||||
@JsonProperty("poster") val poster: String
|
|
||||||
)
|
|
||||||
|
|
||||||
private data class MatchParser(
|
|
||||||
@JsonProperty("linkData") val linkData: List<LinkParser>,
|
|
||||||
@JsonProperty("matchData") val MatchData: MatchDataParser
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
override suspend fun getMainPage(page: Int, request: MainPageRequest): HomePageResponse {
|
override suspend fun getMainPage(page: Int, request: MainPageRequest): HomePageResponse {
|
||||||
val document = app.get(mainUrl).document
|
val document = app.get(mainUrl, interceptor = interceptor).document
|
||||||
val sections = document.select("div.panel")
|
val sections = document.select("div.panel").groupBy { it.selectFirst("h4 b")?.text() }.values
|
||||||
if (sections.isEmpty()) throw ErrorLoadingException()
|
if (sections.isEmpty()) throw ErrorLoadingException()
|
||||||
|
val prova = sections.map {elements ->
|
||||||
|
val home = elements.mapNotNull { it.toMainPageResult() }
|
||||||
|
HomePageList(elements.first()?.selectFirst("h4 b")?.text()?:"Altro", home)
|
||||||
|
}
|
||||||
|
return HomePageResponse(prova)
|
||||||
|
}
|
||||||
|
|
||||||
return HomePageResponse(sections.mapNotNull { sport ->
|
private fun Element.toMainPageResult() : LiveSearchResponse {
|
||||||
val dayMatch = sport.previousElementSiblings().toList().first { it.`is`("h3") }.text()
|
val name = this.selectFirst("b")?.text()?:"Altro"
|
||||||
val categoryName = sport.selectFirst("h4")?.text() ?: "Other"
|
val links = this.select("tr")
|
||||||
|
.toList()
|
||||||
val showsList = sport.select("tr").takeWhile { it.text().contains("Player").not() }
|
.filter { it.hasAttr("class") && it.attr("class") !in listOf("", "audio") }
|
||||||
.filter { it.hasAttr("class") }.drop(1)
|
.map { LinkParser(
|
||||||
|
fixUrl(it.selectFirst("a")?.attr("href")?:""),
|
||||||
val shows = showsList.groupBy { it.text().substringBeforeLast(" ") }.map { matchs ->
|
it.attr("class"),
|
||||||
val posterUrl = fixUrl(
|
it.selectFirst("b")?.text()?:""
|
||||||
sport.selectFirst("h4")?.attr("style")
|
) }
|
||||||
?.substringAfter("(")?.substringBefore(")") ?: ""
|
val dayMatch = this.previousElementSiblings().toList().firstOrNull() { it.`is`("h3") }?.text()
|
||||||
)
|
|
||||||
val hasDate = matchs.key.contains(":")
|
|
||||||
val matchName = if (hasDate) { matchs.key.substringAfter(" ")}
|
|
||||||
else { matchs.key }
|
|
||||||
|
|
||||||
val href = matchs.value.map { match ->
|
|
||||||
val linkUrl = fixUrl(match.selectFirst("a")?.attr("href") ?: return@mapNotNull null)
|
|
||||||
val lang = match.attr("class")
|
|
||||||
LinkParser(linkUrl, lang, matchName)
|
|
||||||
}
|
|
||||||
|
|
||||||
val date = if (hasDate) {
|
|
||||||
dayMatch + " - " + matchs.key.substringBefore(" ")
|
|
||||||
} else {
|
|
||||||
dayMatch
|
|
||||||
}
|
|
||||||
|
|
||||||
LiveSearchResponse(
|
|
||||||
matchName,
|
|
||||||
MatchParser(href, MatchDataParser(date, posterUrl)).toJson(),
|
|
||||||
this@StarLiveProvider.name,
|
|
||||||
TvType.Live,
|
|
||||||
posterUrl,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
HomePageList(
|
|
||||||
categoryName,
|
|
||||||
shows
|
|
||||||
)
|
|
||||||
|
|
||||||
})
|
|
||||||
|
|
||||||
|
val matchData = MatchDataParser(
|
||||||
|
dayMatch?.plus(" - ".plus(this.selectFirst("#evento")?.text()?.substringBefore(" ")))?:"no data",
|
||||||
|
fixUrl(
|
||||||
|
this.selectFirst("h4")?.attr("style")
|
||||||
|
?.substringAfter("(")?.substringBefore(")") ?: ""
|
||||||
|
),
|
||||||
|
this.selectFirst("#evento")?.text()?.substringAfter(" ")?:"Match in $name || $dayMatch"
|
||||||
|
)
|
||||||
|
val href = MatchParser(links, matchData)
|
||||||
|
return LiveSearchResponse(
|
||||||
|
this.selectFirst("#evento")?.text()?.substringAfter(" ")?:"Match in $name",
|
||||||
|
href.toJson(),
|
||||||
|
this@StarLiveProvider.name,
|
||||||
|
TvType.Live,
|
||||||
|
fixUrl(
|
||||||
|
this.selectFirst("h4")?.attr("style")
|
||||||
|
?.substringAfter("(")?.substringBefore(")") ?: ""
|
||||||
|
),
|
||||||
|
posterHeaders = interceptor.getCookieHeaders(mainUrl).toMap()
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
override suspend fun load(url: String): LoadResponse {
|
override suspend fun load(url: String): LoadResponse {
|
||||||
|
@ -88,10 +69,11 @@ class StarLiveProvider : MainAPI() {
|
||||||
return LiveStreamLoadResponse(
|
return LiveStreamLoadResponse(
|
||||||
dataUrl = url,
|
dataUrl = url,
|
||||||
url = matchdata?.linkData?.firstOrNull()?.link ?: mainUrl,
|
url = matchdata?.linkData?.firstOrNull()?.link ?: mainUrl,
|
||||||
name = matchdata?.linkData?.firstOrNull()?.name ?: "No name",
|
name = matchdata?.MatchData?.name ?: "No name",
|
||||||
posterUrl = poster,
|
posterUrl = poster,
|
||||||
plot = matchstart,
|
plot = matchstart,
|
||||||
apiName = this@StarLiveProvider.name
|
apiName = this@StarLiveProvider.name,
|
||||||
|
posterHeaders = interceptor.getCookieHeaders(mainUrl).toMap()
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -99,11 +81,10 @@ class StarLiveProvider : MainAPI() {
|
||||||
data: LinkParser,
|
data: LinkParser,
|
||||||
callback: (ExtractorLink) -> Unit
|
callback: (ExtractorLink) -> Unit
|
||||||
) {
|
) {
|
||||||
val linktoStream = fixUrl(app.get(data.link).document.selectFirst("iframe")!!.attr("src"))
|
val linktoStream = fixUrl(app.get(data.link, interceptor = interceptor).document.selectFirst("iframe")!!.attr("src"))
|
||||||
|
|
||||||
val referrerLink = if (linktoStream.contains("starlive")) {
|
val referrerLink = if (linktoStream.contains("starlive")) {
|
||||||
app.get(linktoStream, referer = data.link).document.selectFirst("iframe")?.attr("src")
|
app.get(linktoStream, referer = data.link, interceptor = interceptor).document.selectFirst("iframe")?.attr("src")
|
||||||
?: return
|
?: linktoStream
|
||||||
} else {
|
} else {
|
||||||
linktoStream
|
linktoStream
|
||||||
}
|
}
|
||||||
|
@ -115,12 +96,18 @@ class StarLiveProvider : MainAPI() {
|
||||||
false -> app.get(linktoStream, referer = data.link).document.select("script")
|
false -> app.get(linktoStream, referer = data.link).document.select("script")
|
||||||
.select("script").toString()
|
.select("script").toString()
|
||||||
}
|
}
|
||||||
val streamUrl = getAndUnpack(packed).substringAfter("var src=\"").substringBefore("\"")
|
var streamUrl = getAndUnpack(packed).substringAfter("var src=\"").substringBefore("\"")
|
||||||
|
if (streamUrl.contains("allowedDomains")){streamUrl = packed.substringAfter("source:'").substringBefore("'")}
|
||||||
|
if (!streamUrl.contains("m3u8")){
|
||||||
|
val script = app.get(linktoStream, referer = data.link, interceptor = interceptor).document.selectFirst("body")?.selectFirst("script").toString()
|
||||||
|
streamUrl = Regex("source: [\\\"'](.*?)[\\\"']").find(script)?.groupValues?.last()?:""
|
||||||
|
}
|
||||||
|
|
||||||
callback(
|
callback(
|
||||||
ExtractorLink(
|
ExtractorLink(
|
||||||
source = this.name,
|
source = this.name,
|
||||||
name = data.name + " - " + data.language,
|
name = data.name + " - " + data.language,
|
||||||
url = streamUrl,
|
url = fixUrl(streamUrl),
|
||||||
quality = Qualities.Unknown.value,
|
quality = Qualities.Unknown.value,
|
||||||
referer = referrerLink,
|
referer = referrerLink,
|
||||||
isM3u8 = true
|
isM3u8 = true
|
||||||
|
@ -136,10 +123,28 @@ class StarLiveProvider : MainAPI() {
|
||||||
subtitleCallback: (SubtitleFile) -> Unit,
|
subtitleCallback: (SubtitleFile) -> Unit,
|
||||||
callback: (ExtractorLink) -> Unit
|
callback: (ExtractorLink) -> Unit
|
||||||
): Boolean {
|
): Boolean {
|
||||||
tryParseJson<MatchParser>(data)?.linkData?.map { link ->
|
tryParseJson<MatchParser>(data)?.linkData?.apmap { link ->
|
||||||
extractVideoLinks(link, callback)
|
extractVideoLinks(link, callback)
|
||||||
}
|
}
|
||||||
|
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
private data class LinkParser(
|
||||||
|
@JsonProperty("link") val link: String,
|
||||||
|
@JsonProperty("lang") val language: String,
|
||||||
|
@JsonProperty("name") val name: String
|
||||||
|
)
|
||||||
|
|
||||||
|
private data class MatchDataParser(
|
||||||
|
@JsonProperty("time") val time: String,
|
||||||
|
@JsonProperty("poster") val poster: String,
|
||||||
|
@JsonProperty("name") val name: String
|
||||||
|
)
|
||||||
|
|
||||||
|
private data class MatchParser(
|
||||||
|
@JsonProperty("linkData") val linkData: List<LinkParser>,
|
||||||
|
@JsonProperty("matchData") val MatchData: MatchDataParser
|
||||||
|
)
|
||||||
|
|
||||||
|
}
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
// use an integer for version numbers
|
// use an integer for version numbers
|
||||||
version = 4
|
version = 5
|
||||||
|
|
||||||
|
|
||||||
cloudstream {
|
cloudstream {
|
||||||
|
@ -22,5 +22,5 @@ cloudstream {
|
||||||
"Movie",
|
"Movie",
|
||||||
)
|
)
|
||||||
|
|
||||||
iconUrl = "https://www.google.com/s2/favicons?domain=streamingcommunity.best&sz=%size%"
|
iconUrl = "https://www.google.com/s2/favicons?domain=streamingcommunity.online&sz=%size%"
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,9 +7,241 @@ import com.lagradost.cloudstream3.LoadResponse.Companion.addTrailer
|
||||||
import com.lagradost.cloudstream3.utils.*
|
import com.lagradost.cloudstream3.utils.*
|
||||||
import com.lagradost.cloudstream3.*
|
import com.lagradost.cloudstream3.*
|
||||||
import org.json.JSONObject
|
import org.json.JSONObject
|
||||||
import java.net.URI
|
|
||||||
import java.security.MessageDigest
|
import java.security.MessageDigest
|
||||||
|
|
||||||
|
class StreamingcommunityProvider: MainAPI() {
|
||||||
|
override var lang = "it"
|
||||||
|
override var mainUrl = "https://streamingcommunity.online"
|
||||||
|
override var name = "StreamingCommunity"
|
||||||
|
override val hasMainPage = true
|
||||||
|
override val hasChromecastSupport = true
|
||||||
|
override val supportedTypes = setOf(
|
||||||
|
TvType.Movie,
|
||||||
|
TvType.TvSeries,
|
||||||
|
)
|
||||||
|
private val userAgent = "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/109.0.0.0 Safari/537.36"
|
||||||
|
|
||||||
|
private fun translateNumber(num: Int): Int? {
|
||||||
|
return when (num) {
|
||||||
|
67 -> 1
|
||||||
|
71 -> 2
|
||||||
|
72 -> 3
|
||||||
|
73 -> 4
|
||||||
|
74 -> 5
|
||||||
|
75 -> 6
|
||||||
|
76 -> 7
|
||||||
|
77 -> 8
|
||||||
|
78 -> 9
|
||||||
|
79 -> 10
|
||||||
|
133 -> 11
|
||||||
|
else -> null
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun translateIp(num: Int): String? {
|
||||||
|
return when (num) {
|
||||||
|
16 -> "sc-b1-01.scws-content.net"
|
||||||
|
17 -> "sc-b1-02.scws-content.net"
|
||||||
|
18 -> "sc-b1-03.scws-content.net"
|
||||||
|
85 -> "sc-b1-04.scws-content.net"
|
||||||
|
95 -> "sc-b1-05.scws-content.net"
|
||||||
|
117 -> "sc-b1-06.scws-content.net"
|
||||||
|
141 -> "sc-b1-07.scws-content.net"
|
||||||
|
142 -> "sc-b1-08.scws-content.net"
|
||||||
|
143 -> "sc-b1-09.scws-content.net"
|
||||||
|
144 -> "sc-b1-10.scws-content.net"
|
||||||
|
else -> null
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
override suspend fun getMainPage(page: Int, request : MainPageRequest): HomePageResponse {
|
||||||
|
val document = app.get(mainUrl, headers = mapOf("user-agent" to userAgent)).document
|
||||||
|
val items = document.select("slider-title").subList(0, 3).map {
|
||||||
|
val films = it.attr("titles-json")
|
||||||
|
val videoData = parseJson<List<VideoElement>>(films)
|
||||||
|
val searchResponses = videoData.subList(0, 12).apmap { searchr ->
|
||||||
|
searchr.toSearchResponse()
|
||||||
|
}
|
||||||
|
HomePageList(it.attr("slider-name"),searchResponses)
|
||||||
|
}
|
||||||
|
if (items.isEmpty()) throw ErrorLoadingException()
|
||||||
|
return HomePageResponse(items)
|
||||||
|
}
|
||||||
|
|
||||||
|
private suspend fun VideoElement.toSearchResponse() : MovieSearchResponse{
|
||||||
|
val id = this.id
|
||||||
|
val name = this.slug
|
||||||
|
val img = this.images[0].url
|
||||||
|
val number = translateNumber(this.images[0].serverID.toInt())
|
||||||
|
val ip = translateIp(this.images[0].proxyID.toInt())
|
||||||
|
val posterUrl = "https://$ip/images/$number/$img"
|
||||||
|
val videoUrl = "$mainUrl/titles/$id-$name"
|
||||||
|
//posterMap[videourl] = posterurl
|
||||||
|
val data = app.post("$mainUrl/api/titles/preview/$id", referer = mainUrl, headers = mapOf("user-agent" to userAgent)).text
|
||||||
|
val datajs = parseJson<Moviedata>(data)
|
||||||
|
val type: TvType = if (datajs.type == "movie") {
|
||||||
|
TvType.Movie
|
||||||
|
} else {
|
||||||
|
TvType.TvSeries
|
||||||
|
}
|
||||||
|
|
||||||
|
return newMovieSearchResponse(datajs.name, videoUrl, type) {
|
||||||
|
this.posterUrl = posterUrl
|
||||||
|
this.year =
|
||||||
|
datajs.releaseDate.substringBefore("-").filter { it.isDigit() }.toIntOrNull()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
override suspend fun search(query: String): List<SearchResponse> {
|
||||||
|
val queryFormatted = query.replace(" ", "%20")
|
||||||
|
val url = "$mainUrl/search?q=$queryFormatted"
|
||||||
|
val document = app.get(url, headers = mapOf("user-agent" to userAgent)).document
|
||||||
|
|
||||||
|
val films =
|
||||||
|
document.selectFirst("the-search-page")!!.attr("records-json").replace(""", """"""")
|
||||||
|
|
||||||
|
val searchResults = parseJson<List<VideoElement>>(films)
|
||||||
|
return searchResults.apmap { result ->
|
||||||
|
result.toSearchResponse()
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
override suspend fun load(url: String): LoadResponse {
|
||||||
|
|
||||||
|
val document = app.get(url,headers = mapOf("user-agent" to userAgent)).document
|
||||||
|
val poster = Regex("url\\('(.*)'").find(document.selectFirst("div.title-wrap")?.attributes()
|
||||||
|
?.get("style") ?: "")?.groupValues?.last() //posterMap[url]
|
||||||
|
val id = url.substringBefore("-").filter { it.isDigit() }
|
||||||
|
val data = app.post("$mainUrl/api/titles/preview/$id", referer = mainUrl, headers = mapOf("user-agent" to userAgent)).text
|
||||||
|
|
||||||
|
val datajs = parseJson<Moviedata>(data)
|
||||||
|
val type: TvType = if (datajs.type == "movie") {
|
||||||
|
TvType.Movie
|
||||||
|
} else {
|
||||||
|
TvType.TvSeries
|
||||||
|
}
|
||||||
|
val trailerInfoJs = document.select("slider-trailer").attr("videos")
|
||||||
|
val trailerInfo = parseJson<List<TrailerElement>>(trailerInfoJs)
|
||||||
|
val trailerUrl: String? = if (trailerInfo.isNotEmpty()) {
|
||||||
|
"https://www.youtube.com/watch?v=${trailerInfo[0].url}"
|
||||||
|
} else {
|
||||||
|
null
|
||||||
|
}
|
||||||
|
|
||||||
|
val year = datajs.releaseDate.substringBefore("-")
|
||||||
|
val correlates = document.selectFirst("slider-title")!!.attr("titles-json")
|
||||||
|
val correlatesData = parseJson<List<VideoElement>>(correlates)
|
||||||
|
val number : Int = if (correlatesData.size<=15) {correlatesData.size} else correlatesData.size-15
|
||||||
|
|
||||||
|
val correlatesList =correlatesData.take(number).apmap {
|
||||||
|
it.toSearchResponse()
|
||||||
|
}
|
||||||
|
|
||||||
|
if (type == TvType.TvSeries) {
|
||||||
|
|
||||||
|
val name = datajs.name
|
||||||
|
val episodeList = arrayListOf<Episode>()
|
||||||
|
|
||||||
|
val episodes =
|
||||||
|
Html.fromHtml(document.selectFirst("season-select")!!.attr("seasons")).toString()
|
||||||
|
val jsonEpisodes = parseJson<List<Season>>(episodes)
|
||||||
|
|
||||||
|
jsonEpisodes.map { seasons ->
|
||||||
|
val season = seasons.number.toInt()
|
||||||
|
val sid = seasons.title_id
|
||||||
|
val episode = seasons.episodes
|
||||||
|
episode.map { ep ->
|
||||||
|
val href = "$mainUrl/watch/$sid?e=${ep.id}"
|
||||||
|
val postImage = if (ep.images.isNotEmpty()) {
|
||||||
|
ep.images.first().originalURL
|
||||||
|
} else {
|
||||||
|
""
|
||||||
|
}
|
||||||
|
episodeList.add(
|
||||||
|
|
||||||
|
newEpisode(href) {
|
||||||
|
this.name = ep.name
|
||||||
|
this.season = season
|
||||||
|
this.episode = ep.number.toInt()
|
||||||
|
this.description = ep.plot
|
||||||
|
this.posterUrl = postImage
|
||||||
|
}
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
if (episodeList.isEmpty()) throw ErrorLoadingException("No Seasons Found")
|
||||||
|
|
||||||
|
return newTvSeriesLoadResponse(name, url, type, episodeList) {
|
||||||
|
this.posterUrl = poster
|
||||||
|
this.year = year.filter { it.isDigit() }.toInt()
|
||||||
|
this.plot = document.selectFirst("div.plot-wrap > p")!!.text()
|
||||||
|
this.duration = datajs.runtime?.toInt()
|
||||||
|
this.rating = (datajs.votes[0].average.toFloatOrNull()?.times(1000))?.toInt()
|
||||||
|
this.tags = datajs.genres.map { it.name }
|
||||||
|
addTrailer(trailerUrl)
|
||||||
|
this.recommendations = correlatesList
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
} else {
|
||||||
|
|
||||||
|
return newMovieLoadResponse(
|
||||||
|
document.selectFirst("div > div > h1")!!.text(),
|
||||||
|
document.select("a.play-hitzone").attr("href"),
|
||||||
|
type,
|
||||||
|
document.select("a.play-hitzone").attr("href")
|
||||||
|
) {
|
||||||
|
posterUrl = fixUrlNull(poster)
|
||||||
|
this.year = year.filter { it.isDigit() }.toInt()
|
||||||
|
this.plot = document.selectFirst("p.plot")!!.text()
|
||||||
|
this.rating = datajs.votes[0].average.toFloatOrNull()?.times(1000)?.toInt()
|
||||||
|
this.tags = datajs.genres.map { it.name }
|
||||||
|
this.duration = datajs.runtime?.toInt()
|
||||||
|
addTrailer(trailerUrl)
|
||||||
|
this.recommendations = correlatesList
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
override suspend fun loadLinks(
|
||||||
|
data: String,
|
||||||
|
isCasting: Boolean,
|
||||||
|
subtitleCallback: (SubtitleFile) -> Unit,
|
||||||
|
callback: (ExtractorLink) -> Unit
|
||||||
|
): Boolean {
|
||||||
|
val ip = app.get("https://api.ipify.org/").text
|
||||||
|
val videosPage = app.get(data, headers = mapOf("user-agent" to userAgent)).document
|
||||||
|
val scwsidJs = videosPage.select("video-player").attr("response").replace(""", """"""")
|
||||||
|
val jsn = JSONObject(scwsidJs)
|
||||||
|
val scwsid = jsn.getString("scws_id")
|
||||||
|
val expire = (System.currentTimeMillis() / 1000 + 172800).toString()
|
||||||
|
|
||||||
|
val token0 = "$expire$ip Yc8U6r8KjAKAepEA".toByteArray()
|
||||||
|
val token1 = MessageDigest.getInstance("MD5").digest(token0)
|
||||||
|
val token2 = base64Encode(token1)
|
||||||
|
val token = token2.replace("=", "").replace("+", "-").replace("/", "_")
|
||||||
|
|
||||||
|
val link = "https://scws.work/master/$scwsid?token=$token&expires=$expire&n=1"
|
||||||
|
|
||||||
|
callback.invoke(
|
||||||
|
ExtractorLink(
|
||||||
|
name,
|
||||||
|
name,
|
||||||
|
link,
|
||||||
|
isM3u8 = true,
|
||||||
|
referer = mainUrl,
|
||||||
|
quality = Qualities.Unknown.value
|
||||||
|
)
|
||||||
|
)
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
data class Moviedata(
|
data class Moviedata(
|
||||||
@JsonProperty("id") val id: Long,
|
@JsonProperty("id") val id: Long,
|
||||||
@JsonProperty("name") val name: String,
|
@JsonProperty("name") val name: String,
|
||||||
|
@ -20,30 +252,25 @@ data class Moviedata(
|
||||||
@JsonProperty("votes") val votes: List<Vote>,
|
@JsonProperty("votes") val votes: List<Vote>,
|
||||||
@JsonProperty("runtime") val runtime: Long? = null
|
@JsonProperty("runtime") val runtime: Long? = null
|
||||||
)
|
)
|
||||||
|
|
||||||
data class Genre(
|
data class Genre(
|
||||||
@JsonProperty("name") val name: String,
|
@JsonProperty("name") val name: String,
|
||||||
@JsonProperty("pivot") val pivot: Pivot,
|
@JsonProperty("pivot") val pivot: Pivot,
|
||||||
)
|
)
|
||||||
|
|
||||||
data class Pivot(
|
data class Pivot(
|
||||||
@JsonProperty("titleID") val titleID: Long,
|
@JsonProperty("titleID") val titleID: Long,
|
||||||
@JsonProperty("genreID") val genreID: Long,
|
@JsonProperty("genreID") val genreID: Long,
|
||||||
)
|
)
|
||||||
|
|
||||||
data class Vote(
|
data class Vote(
|
||||||
@JsonProperty("title_id") val title_id: Long,
|
@JsonProperty("title_id") val title_id: Long,
|
||||||
@JsonProperty("average") val average: String,
|
@JsonProperty("average") val average: String,
|
||||||
@JsonProperty("count") val count: Long,
|
@JsonProperty("count") val count: Long,
|
||||||
@JsonProperty("type") val type: String,
|
@JsonProperty("type") val type: String,
|
||||||
)
|
)
|
||||||
|
|
||||||
data class VideoElement(
|
data class VideoElement(
|
||||||
@JsonProperty("id") val id: Long,
|
@JsonProperty("id") val id: Long,
|
||||||
@JsonProperty("slug") val slug: String,
|
@JsonProperty("slug") val slug: String,
|
||||||
@JsonProperty("images") val images: List<Image>,
|
@JsonProperty("images") val images: List<Image>,
|
||||||
)
|
)
|
||||||
|
|
||||||
data class Image(
|
data class Image(
|
||||||
@JsonProperty("imageable_id") val imageableID: Long,
|
@JsonProperty("imageable_id") val imageableID: Long,
|
||||||
@JsonProperty("imageable_type") val imageableType: String,
|
@JsonProperty("imageable_type") val imageableType: String,
|
||||||
|
@ -55,9 +282,7 @@ data class Image(
|
||||||
// @JsonProperty("proxy") val proxy: Proxy,
|
// @JsonProperty("proxy") val proxy: Proxy,
|
||||||
// @JsonProperty("server") val server: Proxy
|
// @JsonProperty("server") val server: Proxy
|
||||||
)
|
)
|
||||||
|
|
||||||
// Proxy is not used and crashes otherwise
|
// Proxy is not used and crashes otherwise
|
||||||
|
|
||||||
//data class Proxy(
|
//data class Proxy(
|
||||||
// @JsonProperty("id") val id: Long,
|
// @JsonProperty("id") val id: Long,
|
||||||
// @JsonProperty("type") val type: String,
|
// @JsonProperty("type") val type: String,
|
||||||
|
@ -84,7 +309,6 @@ data class Season(
|
||||||
@JsonProperty("updated_at") val updatedAt: String? = "",
|
@JsonProperty("updated_at") val updatedAt: String? = "",
|
||||||
@JsonProperty("episodes") val episodes: List<Episodejson>
|
@JsonProperty("episodes") val episodes: List<Episodejson>
|
||||||
)
|
)
|
||||||
|
|
||||||
data class Episodejson(
|
data class Episodejson(
|
||||||
@JsonProperty("id") val id: Long,
|
@JsonProperty("id") val id: Long,
|
||||||
@JsonProperty("number") val number: Long,
|
@JsonProperty("number") val number: Long,
|
||||||
|
@ -93,7 +317,6 @@ data class Episodejson(
|
||||||
@JsonProperty("season_id") val seasonID: Long,
|
@JsonProperty("season_id") val seasonID: Long,
|
||||||
@JsonProperty("images") val images: List<ImageSeason>
|
@JsonProperty("images") val images: List<ImageSeason>
|
||||||
)
|
)
|
||||||
|
|
||||||
data class ImageSeason(
|
data class ImageSeason(
|
||||||
@JsonProperty("imageable_id") val imageableID: Long,
|
@JsonProperty("imageable_id") val imageableID: Long,
|
||||||
@JsonProperty("imageable_type") val imageableType: String,
|
@JsonProperty("imageable_type") val imageableType: String,
|
||||||
|
@ -103,7 +326,6 @@ data class ImageSeason(
|
||||||
@JsonProperty("type") val type: String,
|
@JsonProperty("type") val type: String,
|
||||||
@JsonProperty("original_url") val originalURL: String
|
@JsonProperty("original_url") val originalURL: String
|
||||||
)
|
)
|
||||||
|
|
||||||
data class TrailerElement(
|
data class TrailerElement(
|
||||||
@JsonProperty("id") val id: Long? = null,
|
@JsonProperty("id") val id: Long? = null,
|
||||||
@JsonProperty("url") val url: String? = null,
|
@JsonProperty("url") val url: String? = null,
|
||||||
|
@ -123,326 +345,4 @@ data class TrailerElement(
|
||||||
@JsonProperty("proxy_id") val proxyID: Any? = null,
|
@JsonProperty("proxy_id") val proxyID: Any? = null,
|
||||||
@JsonProperty("proxy_default_id") val proxyDefaultID: Any? = null,
|
@JsonProperty("proxy_default_id") val proxyDefaultID: Any? = null,
|
||||||
@JsonProperty("scws_id") val scwsID: Any? = null
|
@JsonProperty("scws_id") val scwsID: Any? = null
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
class StreamingcommunityProvider: MainAPI() {
|
|
||||||
override var lang = "it"
|
|
||||||
override var mainUrl = "https://streamingcommunity.cheap"
|
|
||||||
override var name = "StreamingCommunity"
|
|
||||||
override val hasMainPage = true
|
|
||||||
override val hasChromecastSupport = true
|
|
||||||
override val supportedTypes = setOf(
|
|
||||||
TvType.Movie,
|
|
||||||
TvType.TvSeries,
|
|
||||||
)
|
|
||||||
|
|
||||||
private fun translatenumber(num: Int): Int? {
|
|
||||||
return when (num) {
|
|
||||||
67 -> 1
|
|
||||||
71 -> 2
|
|
||||||
72 -> 3
|
|
||||||
73 -> 4
|
|
||||||
74 -> 5
|
|
||||||
75 -> 6
|
|
||||||
76 -> 7
|
|
||||||
77 -> 8
|
|
||||||
78 -> 9
|
|
||||||
79 -> 10
|
|
||||||
133 -> 11
|
|
||||||
else -> null
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun translateip(num: Int): String? {
|
|
||||||
return when (num) {
|
|
||||||
16 -> "sc-b1-01.scws-content.net"
|
|
||||||
17 -> "sc-b1-02.scws-content.net"
|
|
||||||
18 -> "sc-b1-03.scws-content.net"
|
|
||||||
85 -> "sc-b1-04.scws-content.net"
|
|
||||||
95 -> "sc-b1-05.scws-content.net"
|
|
||||||
117 -> "sc-b1-06.scws-content.net"
|
|
||||||
141 -> "sc-b1-07.scws-content.net"
|
|
||||||
142 -> "sc-b1-08.scws-content.net"
|
|
||||||
143 -> "sc-b1-09.scws-content.net"
|
|
||||||
144 -> "sc-b1-10.scws-content.net"
|
|
||||||
else -> null
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// companion object {
|
|
||||||
// val posterMap = hashMapOf<String, String>()
|
|
||||||
// }
|
|
||||||
|
|
||||||
override suspend fun getMainPage(page: Int, request : MainPageRequest): HomePageResponse {
|
|
||||||
val items = ArrayList<HomePageList>()
|
|
||||||
val document = app.get(mainUrl).document
|
|
||||||
document.select("slider-title").subList(0, 3).map { it ->
|
|
||||||
if (it.attr("slider-name") != "In arrivo") {
|
|
||||||
val films = it.attr("titles-json")
|
|
||||||
val lista = mutableListOf<MovieSearchResponse>()
|
|
||||||
val videoData = parseJson<List<VideoElement>>(films)
|
|
||||||
|
|
||||||
videoData.subList(0, 12).apmap { searchr ->
|
|
||||||
val id = searchr.id
|
|
||||||
val name = searchr.slug
|
|
||||||
val img = searchr.images[0].url
|
|
||||||
val number = translatenumber(searchr.images[0].serverID.toInt())
|
|
||||||
val ip = translateip(searchr.images[0].proxyID.toInt())
|
|
||||||
val posterurl = "https://$ip/images/$number/$img"
|
|
||||||
val videourl = "$mainUrl/titles/$id-$name"
|
|
||||||
//posterMap[videourl] = posterurl
|
|
||||||
val data = app.post("$mainUrl/api/titles/preview/$id", referer = mainUrl).text
|
|
||||||
val datajs = parseJson<Moviedata>(data)
|
|
||||||
val type: TvType = if (datajs.type == "movie") {
|
|
||||||
TvType.Movie
|
|
||||||
} else {
|
|
||||||
TvType.TvSeries
|
|
||||||
}
|
|
||||||
|
|
||||||
lista.add(
|
|
||||||
MovieSearchResponse(
|
|
||||||
datajs.name,
|
|
||||||
videourl,
|
|
||||||
this.name,
|
|
||||||
type,
|
|
||||||
posterurl,
|
|
||||||
datajs.releaseDate.substringBefore("-").filter { it.isDigit() }
|
|
||||||
.toIntOrNull(),
|
|
||||||
null,
|
|
||||||
)
|
|
||||||
)
|
|
||||||
}
|
|
||||||
items.add(HomePageList(it.attr("slider-name"), lista))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (items.size <= 0) throw ErrorLoadingException()
|
|
||||||
return HomePageResponse(items)
|
|
||||||
}
|
|
||||||
|
|
||||||
override suspend fun search(query: String): List<SearchResponse> {
|
|
||||||
val queryformatted = query.replace(" ", "%20")
|
|
||||||
val url = "$mainUrl/search?q=$queryformatted"
|
|
||||||
val document = app.get(url).document
|
|
||||||
|
|
||||||
val films =
|
|
||||||
document.selectFirst("the-search-page")!!.attr("records-json").replace(""", """"""")
|
|
||||||
|
|
||||||
val searchresults = parseJson<List<VideoElement>>(films)
|
|
||||||
return searchresults.apmap { result ->
|
|
||||||
val id = result.id
|
|
||||||
val name = result.slug
|
|
||||||
val img = result.images[0].url
|
|
||||||
val number = translatenumber(result.images[0].serverID.toInt())
|
|
||||||
val ip = translateip(result.images[0].proxyID.toInt())
|
|
||||||
val data = app.post("$mainUrl/api/titles/preview/$id", referer = mainUrl).text
|
|
||||||
val datajs = parseJson<Moviedata>(data)
|
|
||||||
val posterurl = "https://$ip/images/$number/$img"
|
|
||||||
val videourl = "$mainUrl/titles/$id-$name"
|
|
||||||
//posterMap[videourl] = posterurl
|
|
||||||
if (datajs.type == "movie") {
|
|
||||||
val type = TvType.Movie
|
|
||||||
MovieSearchResponse(
|
|
||||||
datajs.name,
|
|
||||||
videourl,
|
|
||||||
this.name,
|
|
||||||
type,
|
|
||||||
posterurl,
|
|
||||||
datajs.releaseDate.substringBefore("-").filter { it.isDigit() }.toIntOrNull(),
|
|
||||||
null,
|
|
||||||
)
|
|
||||||
} else {
|
|
||||||
val type = TvType.TvSeries
|
|
||||||
TvSeriesSearchResponse(
|
|
||||||
datajs.name,
|
|
||||||
videourl,
|
|
||||||
this.name,
|
|
||||||
type,
|
|
||||||
posterurl,
|
|
||||||
datajs.releaseDate.substringBefore("-").filter { it.isDigit() }.toIntOrNull(),
|
|
||||||
null,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
override suspend fun load(url: String): LoadResponse {
|
|
||||||
|
|
||||||
val document = app.get(url).document
|
|
||||||
val poster = Regex("url\\('(.*)'").find(document.selectFirst("div.title-wrap")?.attributes()
|
|
||||||
?.get("style") ?: "")?.groupValues?.last() //posterMap[url]
|
|
||||||
val id = url.substringBefore("-").filter { it.isDigit() }
|
|
||||||
val data = app.post("$mainUrl/api/titles/preview/$id", referer = mainUrl).text
|
|
||||||
|
|
||||||
val datajs = parseJson<Moviedata>(data)
|
|
||||||
val type: TvType = if (datajs.type == "movie") {
|
|
||||||
TvType.Movie
|
|
||||||
} else {
|
|
||||||
TvType.TvSeries
|
|
||||||
}
|
|
||||||
val trailerinfojs = document.select("slider-trailer").attr("videos")
|
|
||||||
val trailerinfo = parseJson<List<TrailerElement>>(trailerinfojs)
|
|
||||||
val trailerurl: String? = if (trailerinfo.isNotEmpty()) {
|
|
||||||
"https://www.youtube.com/watch?v=${trailerinfo[0].url}"
|
|
||||||
} else {
|
|
||||||
null
|
|
||||||
}
|
|
||||||
|
|
||||||
val year = datajs.releaseDate.substringBefore("-")
|
|
||||||
|
|
||||||
val correlatijs = document.selectFirst("slider-title")!!.attr("titles-json")
|
|
||||||
val listacorr = mutableListOf<MovieSearchResponse>()
|
|
||||||
val correlatidata = parseJson<List<VideoElement>>(correlatijs)
|
|
||||||
val number : Int = if (correlatidata.size<=15) {correlatidata.size} else correlatidata.size-15
|
|
||||||
|
|
||||||
correlatidata.take(number).apmap { searchr ->
|
|
||||||
val idcorr = searchr.id
|
|
||||||
val name = searchr.slug
|
|
||||||
val img = searchr.images[0].url
|
|
||||||
val number = translatenumber(searchr.images[0].serverID.toInt())
|
|
||||||
val ip = translateip(searchr.images[0].proxyID.toInt())
|
|
||||||
val datacorrel = app.post("$mainUrl/api/titles/preview/$idcorr", referer = mainUrl).text
|
|
||||||
val datajscorrel = parseJson<Moviedata>(datacorrel)
|
|
||||||
val videourl = "$mainUrl/titles/$idcorr-$name"
|
|
||||||
val posterurl = "https://$ip/images/$number/$img"
|
|
||||||
|
|
||||||
//posterMap[videourl] = posterurl
|
|
||||||
val typecorr: TvType = if (datajscorrel.type == "movie") {
|
|
||||||
TvType.Movie
|
|
||||||
} else {
|
|
||||||
TvType.TvSeries
|
|
||||||
}
|
|
||||||
|
|
||||||
listacorr.add(
|
|
||||||
MovieSearchResponse(
|
|
||||||
datajscorrel.name,
|
|
||||||
videourl,
|
|
||||||
this.name,
|
|
||||||
typecorr,
|
|
||||||
posterurl,
|
|
||||||
datajscorrel.releaseDate.substringBefore("-").filter { it.isDigit() }
|
|
||||||
.toIntOrNull(),
|
|
||||||
null,
|
|
||||||
)
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
if (type == TvType.TvSeries) {
|
|
||||||
|
|
||||||
val name = datajs.name
|
|
||||||
val episodeList = arrayListOf<Episode>()
|
|
||||||
|
|
||||||
val episodes =
|
|
||||||
Html.fromHtml(document.selectFirst("season-select")!!.attr("seasons")).toString()
|
|
||||||
val jsonEpisodes = parseJson<List<Season>>(episodes)
|
|
||||||
|
|
||||||
jsonEpisodes.map { seasons ->
|
|
||||||
val stagione = seasons.number.toInt()
|
|
||||||
val sid = seasons.title_id
|
|
||||||
val episodio = seasons.episodes
|
|
||||||
episodio.map { ep ->
|
|
||||||
val href = "$mainUrl/watch/$sid?e=${ep.id}"
|
|
||||||
val postimage = if (ep.images.isNotEmpty()) {
|
|
||||||
ep.images.first().originalURL
|
|
||||||
} else {
|
|
||||||
""
|
|
||||||
}
|
|
||||||
episodeList.add(
|
|
||||||
|
|
||||||
newEpisode(href) {
|
|
||||||
this.name = ep.name
|
|
||||||
this.season = stagione
|
|
||||||
this.episode = ep.number.toInt()
|
|
||||||
this.description = ep.plot
|
|
||||||
this.posterUrl = postimage
|
|
||||||
}
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
if (episodeList.isEmpty()) throw ErrorLoadingException("No Seasons Found")
|
|
||||||
|
|
||||||
return newTvSeriesLoadResponse(name, url, type, episodeList) {
|
|
||||||
this.posterUrl = poster
|
|
||||||
this.year = year.filter { it.isDigit() }.toInt()
|
|
||||||
this.plot = document.selectFirst("div.plot-wrap > p")!!.text()
|
|
||||||
this.duration = datajs.runtime?.toInt()
|
|
||||||
this.rating = (datajs.votes[0].average.toFloatOrNull()?.times(1000))?.toInt()
|
|
||||||
this.tags = datajs.genres.map { it.name }
|
|
||||||
addTrailer(trailerurl)
|
|
||||||
this.recommendations = listacorr
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
} else {
|
|
||||||
|
|
||||||
return newMovieLoadResponse(
|
|
||||||
document.selectFirst("div > div > h1")!!.text(),
|
|
||||||
document.select("a.play-hitzone").attr("href"),
|
|
||||||
type,
|
|
||||||
document.select("a.play-hitzone").attr("href")
|
|
||||||
) {
|
|
||||||
posterUrl = fixUrlNull(poster)
|
|
||||||
this.year = year.filter { it.isDigit() }.toInt()
|
|
||||||
this.plot = document.selectFirst("p.plot")!!.text()
|
|
||||||
this.rating = datajs.votes[0].average.toFloatOrNull()?.times(1000)?.toInt()
|
|
||||||
this.tags = datajs.genres.map { it.name }
|
|
||||||
this.duration = datajs.runtime?.toInt()
|
|
||||||
addTrailer(trailerurl)
|
|
||||||
this.recommendations = listacorr
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
private suspend fun getM3u8Qualities(
|
|
||||||
m3u8Link: String,
|
|
||||||
referer: String,
|
|
||||||
qualityName: String,
|
|
||||||
): List<ExtractorLink> {
|
|
||||||
return M3u8Helper.generateM3u8(
|
|
||||||
this.name,
|
|
||||||
m3u8Link,
|
|
||||||
referer,
|
|
||||||
name = "${this.name} - $qualityName"
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
override suspend fun loadLinks(
|
|
||||||
data: String,
|
|
||||||
isCasting: Boolean,
|
|
||||||
subtitleCallback: (SubtitleFile) -> Unit,
|
|
||||||
callback: (ExtractorLink) -> Unit
|
|
||||||
): Boolean {
|
|
||||||
val ip = app.get("https://api.ipify.org/").text
|
|
||||||
val videors = app.get(data).document
|
|
||||||
val scwsidjs = videors.select("video-player").attr("response").replace(""", """"""")
|
|
||||||
val jsn = JSONObject(scwsidjs)
|
|
||||||
val scwsid = jsn.getString("scws_id")
|
|
||||||
val expire = (System.currentTimeMillis() / 1000 + 172800).toString()
|
|
||||||
|
|
||||||
val token0 = "$expire$ip Yc8U6r8KjAKAepEA".toByteArray()
|
|
||||||
val token1 = MessageDigest.getInstance("MD5").digest(token0)
|
|
||||||
val token2 = base64Encode(token1)
|
|
||||||
val token = token2.replace("=", "").replace("+", "-").replace("/", "_")
|
|
||||||
|
|
||||||
val link = "https://scws.work/master/$scwsid?token=$token&expires=$expire&n=1"
|
|
||||||
|
|
||||||
callback.invoke(
|
|
||||||
ExtractorLink(
|
|
||||||
name,
|
|
||||||
name,
|
|
||||||
link,
|
|
||||||
isM3u8 = true,
|
|
||||||
referer = mainUrl,
|
|
||||||
quality = Qualities.Unknown.value
|
|
||||||
)
|
|
||||||
)
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,5 +1,5 @@
|
||||||
// use an integer for version numbers
|
// use an integer for version numbers
|
||||||
version = 4
|
version = 5
|
||||||
|
|
||||||
|
|
||||||
cloudstream {
|
cloudstream {
|
||||||
|
|
|
@ -9,7 +9,7 @@ import com.lagradost.cloudstream3.network.CloudflareKiller
|
||||||
|
|
||||||
class TantifilmProvider : MainAPI() {
|
class TantifilmProvider : MainAPI() {
|
||||||
override var lang = "it"
|
override var lang = "it"
|
||||||
override var mainUrl = "https://tantifilm.mobi"
|
override var mainUrl = "https://tantifilm.recipes"
|
||||||
override var name = "Tantifilm"
|
override var name = "Tantifilm"
|
||||||
override val hasMainPage = true
|
override val hasMainPage = true
|
||||||
override val hasChromecastSupport = true
|
override val hasChromecastSupport = true
|
||||||
|
|
Loading…
Reference in New Issue