This commit is contained in:
antonydp 2023-01-30 15:30:30 +01:00
commit e7609e8da5
14 changed files with 357 additions and 440 deletions

View File

@ -1,5 +1,5 @@
// use an integer for version numbers
version = 1
version = 2
cloudstream {

View File

@ -2,7 +2,7 @@ package com.lagradost
import com.lagradost.cloudstream3.*
import com.lagradost.cloudstream3.utils.*
import org.jsoup.nodes.Document
class CalcioStreamingProvider : MainAPI() {
override var lang = "it"
@ -58,12 +58,21 @@ class CalcioStreamingProvider : MainAPI() {
poster,
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(
url: String,
@ -71,25 +80,28 @@ class CalcioStreamingProvider : MainAPI() {
) {
val document = app.get(url).document
document.select("button.btn").forEach { button ->
val link1 = button.attr("data-link")
val doc2 = app.get(link1).document
val truelink = doc2.selectFirst("iframe")!!.attr("src")
val newpage = app.get(truelink, referer = link1).document
val streamurl = Regex(""""((.|\n)*?).";""").find(
getAndUnpack(
newpage.select("script")[6].childNode(0).toString()
))!!.value.replace("""src="""", "").replace(""""""", "").replace(";", "")
callback(
ExtractorLink(
this.name,
button.text(),
streamurl,
truelink,
quality = 0,
true
)
)
var link = button.attr("data-link")
var oldLink = link
var videoNotFound = true
while (videoNotFound) {
val doc = app.get(link).document
link = doc.selectFirst("iframe")?.attr("src") ?: break
val newpage = app.get(fixUrl(link), referer = oldLink).document
oldLink = link
if (newpage.select("script").size >= 6 && matchFound(newpage)){
videoNotFound = false
callback(
ExtractorLink(
this.name,
button.text(),
getUrl(newpage),
fixUrl(link),
quality = 0,
true
)
)
}
}
}
}

View File

@ -1,5 +1,5 @@
// use an integer for version numbers
version = 2
version = 3
cloudstream {

View File

@ -10,7 +10,7 @@ import org.jsoup.nodes.Element
class CineBlog01Provider : MainAPI() {
override var lang = "it"
override var mainUrl = "https://www.cineblog01.moe"
override var mainUrl = "https://www.cineblog01.mom"
override var name = "CineBlog01"
override val hasMainPage = true
override val hasChromecastSupport = true

View File

@ -1,5 +1,5 @@
// use an integer for version numbers
version = 2
version = 3
cloudstream {
language = "it"

View File

@ -11,7 +11,7 @@ import org.jsoup.nodes.Element
class EurostreamingProvider : MainAPI() {
override var lang = "it"
override var mainUrl = "https://eurostreaming.taxi"
override var mainUrl = "https://eurostreaming.expert"
override var name = "Eurostreaming"
override val hasMainPage = true
override val hasChromecastSupport = true

View File

@ -1,5 +1,5 @@
// use an integer for version numbers
version = 4
version = 5
cloudstream {

View File

@ -16,7 +16,7 @@ import com.lagradost.cloudstream3.network.CloudflareKiller
class FilmpertuttiProvider : MainAPI() {
override var lang = "it"
override var mainUrl = "https://filmpertutti.skin"
override var mainUrl = "https://filmpertutti.tips"
override var name = "FilmPerTutti"
override val hasMainPage = true
override val hasChromecastSupport = true

View File

@ -1,5 +1,5 @@
// use an integer for version numbers
version = 1
version = 2
cloudstream {

View File

@ -2,9 +2,11 @@ package com.lagradost
import com.fasterxml.jackson.annotation.JsonProperty
import com.lagradost.cloudstream3.*
import com.lagradost.cloudstream3.network.CloudflareKiller
import com.lagradost.cloudstream3.utils.*
import com.lagradost.cloudstream3.utils.AppUtils.toJson
import com.lagradost.cloudstream3.utils.AppUtils.tryParseJson
import org.jsoup.nodes.Element
class StarLiveProvider : MainAPI() {
override var mainUrl = "https://starlive.xyz"
@ -13,72 +15,51 @@ class StarLiveProvider : MainAPI() {
override var lang = "it"
override val hasChromecastSupport = true
override val supportedTypes = setOf(TvType.Live)
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
)
private val interceptor = CloudflareKiller()
override suspend fun getMainPage(page: Int, request: MainPageRequest): HomePageResponse {
val document = app.get(mainUrl).document
val sections = document.select("div.panel")
val document = app.get(mainUrl, interceptor = interceptor).document
val sections = document.select("div.panel").groupBy { it.selectFirst("h4 b")?.text() }.values
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 ->
val dayMatch = sport.previousElementSiblings().toList().first { it.`is`("h3") }.text()
val categoryName = sport.selectFirst("h4")?.text() ?: "Other"
val showsList = sport.select("tr").takeWhile { it.text().contains("Player").not() }
.filter { it.hasAttr("class") }.drop(1)
val shows = showsList.groupBy { it.text().substringBeforeLast(" ") }.map { matchs ->
val posterUrl = fixUrl(
sport.selectFirst("h4")?.attr("style")
?.substringAfter("(")?.substringBefore(")") ?: ""
)
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
)
})
private fun Element.toMainPageResult() : LiveSearchResponse {
val name = this.selectFirst("b")?.text()?:"Altro"
val links = this.select("tr")
.toList()
.filter { it.hasAttr("class") && it.attr("class") !in listOf("", "audio") }
.map { LinkParser(
fixUrl(it.selectFirst("a")?.attr("href")?:""),
it.attr("class"),
it.selectFirst("b")?.text()?:""
) }
val dayMatch = this.previousElementSiblings().toList().firstOrNull() { it.`is`("h3") }?.text()
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 {
@ -88,10 +69,11 @@ class StarLiveProvider : MainAPI() {
return LiveStreamLoadResponse(
dataUrl = url,
url = matchdata?.linkData?.firstOrNull()?.link ?: mainUrl,
name = matchdata?.linkData?.firstOrNull()?.name ?: "No name",
name = matchdata?.MatchData?.name ?: "No name",
posterUrl = poster,
plot = matchstart,
apiName = this@StarLiveProvider.name
apiName = this@StarLiveProvider.name,
posterHeaders = interceptor.getCookieHeaders(mainUrl).toMap()
)
}
@ -99,11 +81,10 @@ class StarLiveProvider : MainAPI() {
data: LinkParser,
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")) {
app.get(linktoStream, referer = data.link).document.selectFirst("iframe")?.attr("src")
?: return
app.get(linktoStream, referer = data.link, interceptor = interceptor).document.selectFirst("iframe")?.attr("src")
?: linktoStream
} else {
linktoStream
}
@ -115,12 +96,18 @@ class StarLiveProvider : MainAPI() {
false -> app.get(linktoStream, referer = data.link).document.select("script")
.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(
ExtractorLink(
source = this.name,
name = data.name + " - " + data.language,
url = streamUrl,
url = fixUrl(streamUrl),
quality = Qualities.Unknown.value,
referer = referrerLink,
isM3u8 = true
@ -136,10 +123,28 @@ class StarLiveProvider : MainAPI() {
subtitleCallback: (SubtitleFile) -> Unit,
callback: (ExtractorLink) -> Unit
): Boolean {
tryParseJson<MatchParser>(data)?.linkData?.map { link ->
tryParseJson<MatchParser>(data)?.linkData?.apmap { link ->
extractVideoLinks(link, callback)
}
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
)
}

View File

@ -1,5 +1,5 @@
// use an integer for version numbers
version = 4
version = 5
cloudstream {
@ -22,5 +22,5 @@ cloudstream {
"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%"
}

View File

@ -7,9 +7,241 @@ import com.lagradost.cloudstream3.LoadResponse.Companion.addTrailer
import com.lagradost.cloudstream3.utils.*
import com.lagradost.cloudstream3.*
import org.json.JSONObject
import java.net.URI
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("&quot;", """"""")
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("&quot;", """"""")
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(
@JsonProperty("id") val id: Long,
@JsonProperty("name") val name: String,
@ -20,30 +252,25 @@ data class Moviedata(
@JsonProperty("votes") val votes: List<Vote>,
@JsonProperty("runtime") val runtime: Long? = null
)
data class Genre(
@JsonProperty("name") val name: String,
@JsonProperty("pivot") val pivot: Pivot,
)
data class Pivot(
@JsonProperty("titleID") val titleID: Long,
@JsonProperty("genreID") val genreID: Long,
)
data class Vote(
@JsonProperty("title_id") val title_id: Long,
@JsonProperty("average") val average: String,
@JsonProperty("count") val count: Long,
@JsonProperty("type") val type: String,
)
data class VideoElement(
@JsonProperty("id") val id: Long,
@JsonProperty("slug") val slug: String,
@JsonProperty("images") val images: List<Image>,
)
data class Image(
@JsonProperty("imageable_id") val imageableID: Long,
@JsonProperty("imageable_type") val imageableType: String,
@ -55,9 +282,7 @@ data class Image(
// @JsonProperty("proxy") val proxy: Proxy,
// @JsonProperty("server") val server: Proxy
)
// Proxy is not used and crashes otherwise
//data class Proxy(
// @JsonProperty("id") val id: Long,
// @JsonProperty("type") val type: String,
@ -84,7 +309,6 @@ data class Season(
@JsonProperty("updated_at") val updatedAt: String? = "",
@JsonProperty("episodes") val episodes: List<Episodejson>
)
data class Episodejson(
@JsonProperty("id") val id: Long,
@JsonProperty("number") val number: Long,
@ -93,7 +317,6 @@ data class Episodejson(
@JsonProperty("season_id") val seasonID: Long,
@JsonProperty("images") val images: List<ImageSeason>
)
data class ImageSeason(
@JsonProperty("imageable_id") val imageableID: Long,
@JsonProperty("imageable_type") val imageableType: String,
@ -103,7 +326,6 @@ data class ImageSeason(
@JsonProperty("type") val type: String,
@JsonProperty("original_url") val originalURL: String
)
data class TrailerElement(
@JsonProperty("id") val id: Long? = null,
@JsonProperty("url") val url: String? = null,
@ -123,326 +345,4 @@ data class TrailerElement(
@JsonProperty("proxy_id") val proxyID: Any? = null,
@JsonProperty("proxy_default_id") val proxyDefaultID: 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("&quot;", """"""")
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("&quot;", """"""")
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
}
}
)

View File

@ -1,5 +1,5 @@
// use an integer for version numbers
version = 4
version = 5
cloudstream {

View File

@ -9,7 +9,7 @@ import com.lagradost.cloudstream3.network.CloudflareKiller
class TantifilmProvider : MainAPI() {
override var lang = "it"
override var mainUrl = "https://tantifilm.mobi"
override var mainUrl = "https://tantifilm.recipes"
override var name = "Tantifilm"
override val hasMainPage = true
override val hasChromecastSupport = true