mirror of
https://github.com/hexated/cloudstream-extensions-hexated.git
synced 2024-08-15 00:03:22 +00:00
369 lines
14 KiB
Kotlin
369 lines
14 KiB
Kotlin
package com.hexated
|
|
|
|
import com.fasterxml.jackson.annotation.JsonProperty
|
|
import com.hexated.SubsExtractors.invokeOpenSubs
|
|
import com.hexated.SubsExtractors.invokeWatchsomuch
|
|
import com.lagradost.cloudstream3.*
|
|
import com.lagradost.cloudstream3.LoadResponse.Companion.addActors
|
|
import com.lagradost.cloudstream3.LoadResponse.Companion.addImdbId
|
|
import com.lagradost.cloudstream3.LoadResponse.Companion.addTrailer
|
|
import com.lagradost.cloudstream3.utils.*
|
|
import com.lagradost.cloudstream3.utils.AppUtils.parseJson
|
|
import com.lagradost.cloudstream3.utils.AppUtils.toJson
|
|
|
|
class StremioC : MainAPI() {
|
|
override var mainUrl = "https://stremio.github.io/stremio-static-addon-example"
|
|
override var name = "StremioC"
|
|
override val supportedTypes = setOf(TvType.Others)
|
|
override val hasMainPage = true
|
|
|
|
companion object {
|
|
private const val cinemataUrl = "https://v3-cinemeta.strem.io"
|
|
private const val TRACKER_LIST_URL = "https://raw.githubusercontent.com/ngosang/trackerslist/master/trackers_best.txt"
|
|
}
|
|
|
|
override suspend fun getMainPage(page: Int, request: MainPageRequest): HomePageResponse {
|
|
mainUrl = mainUrl.fixSourceUrl()
|
|
val res = app.get("${mainUrl}/manifest.json").parsedSafe<Manifest>()
|
|
val lists = mutableListOf<HomePageList>()
|
|
res?.catalogs?.apmap { catalog ->
|
|
catalog.toHomePageList(this).let {
|
|
if (it.list.isNotEmpty()) lists.add(it)
|
|
}
|
|
}
|
|
return HomePageResponse(
|
|
lists,
|
|
false
|
|
)
|
|
}
|
|
|
|
override suspend fun search(query: String): List<SearchResponse> {
|
|
mainUrl = mainUrl.fixSourceUrl()
|
|
val res = app.get("${mainUrl}/manifest.json").parsedSafe<Manifest>()
|
|
val list = mutableListOf<SearchResponse>()
|
|
res?.catalogs?.apmap { catalog ->
|
|
list.addAll(catalog.search(query, this))
|
|
}
|
|
return list.distinct()
|
|
}
|
|
|
|
override suspend fun load(url: String): LoadResponse {
|
|
val res = parseJson<CatalogEntry>(url)
|
|
mainUrl =
|
|
if ((res.type == "movie" || res.type == "series") && isImdborTmdb(res.id)) cinemataUrl else mainUrl
|
|
val json = app.get("${mainUrl}/meta/${res.type}/${res.id}.json")
|
|
.parsedSafe<CatalogResponse>()?.meta ?: throw RuntimeException(url)
|
|
return json.toLoadResponse(this, res.id)
|
|
}
|
|
|
|
override suspend fun loadLinks(
|
|
data: String,
|
|
isCasting: Boolean,
|
|
subtitleCallback: (SubtitleFile) -> Unit,
|
|
callback: (ExtractorLink) -> Unit
|
|
): Boolean {
|
|
val loadData = parseJson<LoadData>(data)
|
|
val request = app.get(
|
|
"${mainUrl}/stream/${loadData.type}/${loadData.id}.json",
|
|
timeout = 120L
|
|
)
|
|
if (request.isSuccessful) {
|
|
val res = request.parsedSafe<StreamsResponse>()
|
|
res?.streams?.forEach { stream ->
|
|
stream.runCallback(subtitleCallback, callback)
|
|
}
|
|
} else {
|
|
argamap(
|
|
{
|
|
invokeStremioX(loadData.type, loadData.id, subtitleCallback, callback)
|
|
},
|
|
{
|
|
invokeWatchsomuch(
|
|
loadData.imdbId,
|
|
loadData.season,
|
|
loadData.episode,
|
|
subtitleCallback
|
|
)
|
|
},
|
|
{
|
|
invokeOpenSubs(
|
|
loadData.imdbId,
|
|
loadData.season,
|
|
loadData.episode,
|
|
subtitleCallback
|
|
)
|
|
},
|
|
)
|
|
}
|
|
|
|
return true
|
|
}
|
|
|
|
private suspend fun invokeStremioX(
|
|
type: String?,
|
|
id: String?,
|
|
subtitleCallback: (SubtitleFile) -> Unit,
|
|
callback: (ExtractorLink) -> Unit
|
|
) {
|
|
val sites = AcraApplication.getKey<Array<CustomSite>>(USER_PROVIDER_API)?.toMutableList()
|
|
?: mutableListOf()
|
|
sites.filter { it.parentJavaClass == "StremioX" }.apmap { site ->
|
|
val res = app.get(
|
|
"${site.url.fixSourceUrl()}/stream/${type}/${id}.json",
|
|
timeout = 120L
|
|
).parsedSafe<StreamsResponse>()
|
|
res?.streams?.forEach { stream ->
|
|
stream.runCallback(subtitleCallback, callback)
|
|
}
|
|
}
|
|
}
|
|
|
|
data class LoadData(
|
|
val type: String? = null,
|
|
val id: String? = null,
|
|
val season: Int? = null,
|
|
val episode: Int? = null,
|
|
val imdbId: String? = null,
|
|
)
|
|
|
|
data class CustomSite(
|
|
@JsonProperty("parentJavaClass") val parentJavaClass: String,
|
|
@JsonProperty("name") val name: String,
|
|
@JsonProperty("url") val url: String,
|
|
@JsonProperty("lang") val lang: String,
|
|
)
|
|
|
|
// check if id is imdb/tmdb cause stremio addons like torrentio works base on imdbId
|
|
private fun isImdborTmdb(url: String?): Boolean {
|
|
return imdbUrlToIdNullable(url) != null || url?.startsWith("tmdb:") == true
|
|
}
|
|
|
|
private data class Manifest(val catalogs: List<Catalog>)
|
|
private data class Catalog(
|
|
var name: String?,
|
|
val id: String,
|
|
val type: String?,
|
|
val types: MutableList<String> = mutableListOf()
|
|
) {
|
|
init {
|
|
if (type != null) types.add(type)
|
|
}
|
|
|
|
suspend fun search(query: String, provider: StremioC): List<SearchResponse> {
|
|
val entries = mutableListOf<SearchResponse>()
|
|
types.forEach { type ->
|
|
val res = app.get(
|
|
"${provider.mainUrl}/catalog/${type}/${id}/search=${query}.json",
|
|
timeout = 120L
|
|
).parsedSafe<CatalogResponse>()
|
|
res?.metas?.forEach { entry ->
|
|
entries.add(entry.toSearchResponse(provider))
|
|
}
|
|
}
|
|
return entries
|
|
}
|
|
|
|
suspend fun toHomePageList(provider: StremioC): HomePageList {
|
|
val entries = mutableListOf<SearchResponse>()
|
|
types.forEach { type ->
|
|
val res = app.get(
|
|
"${provider.mainUrl}/catalog/${type}/${id}.json",
|
|
timeout = 120L
|
|
).parsedSafe<CatalogResponse>()
|
|
res?.metas?.forEach { entry ->
|
|
entries.add(entry.toSearchResponse(provider))
|
|
}
|
|
}
|
|
return HomePageList(
|
|
"$type - ${name ?: id}",
|
|
entries
|
|
)
|
|
}
|
|
}
|
|
|
|
private data class CatalogResponse(val metas: List<CatalogEntry>?, val meta: CatalogEntry?)
|
|
|
|
private data class Trailer(
|
|
val source: String?,
|
|
val type: String?
|
|
)
|
|
|
|
private data class CatalogEntry(
|
|
@JsonProperty("name") val name: String,
|
|
@JsonProperty("id") val id: String,
|
|
@JsonProperty("poster") val poster: String?,
|
|
@JsonProperty("background") val background: String?,
|
|
@JsonProperty("description") val description: String?,
|
|
@JsonProperty("imdbRating") val imdbRating: String?,
|
|
@JsonProperty("type") val type: String?,
|
|
@JsonProperty("videos") val videos: List<Video>?,
|
|
@JsonProperty("genre") val genre: List<String>?,
|
|
@JsonProperty("genres") val genres: List<String>?,
|
|
@JsonProperty("cast") val cast: List<String>?,
|
|
@JsonProperty("year") val yearNum: String? = null,
|
|
@JsonProperty("trailers") val trailersSources: ArrayList<Trailer>? = arrayListOf()
|
|
) {
|
|
fun toSearchResponse(provider: StremioC): SearchResponse {
|
|
return provider.newMovieSearchResponse(
|
|
fixTitle(name),
|
|
this.toJson(),
|
|
TvType.Others
|
|
) {
|
|
posterUrl = poster
|
|
}
|
|
}
|
|
|
|
suspend fun toLoadResponse(provider: StremioC, imdbId: String?): LoadResponse {
|
|
if (videos.isNullOrEmpty()) {
|
|
return provider.newMovieLoadResponse(
|
|
name,
|
|
"${provider.mainUrl}/meta/${type}/${id}.json",
|
|
TvType.Movie,
|
|
LoadData(type, id, imdbId = imdbId)
|
|
) {
|
|
posterUrl = poster
|
|
backgroundPosterUrl = background
|
|
rating = imdbRating.toRatingInt()
|
|
plot = description
|
|
year = yearNum?.toIntOrNull()
|
|
tags = genre ?: genres
|
|
addActors(cast)
|
|
addTrailer(trailersSources?.map { "https://www.youtube.com/watch?v=${it.source}" })
|
|
addImdbId(imdbId)
|
|
}
|
|
} else {
|
|
return provider.newTvSeriesLoadResponse(
|
|
name,
|
|
"${provider.mainUrl}/meta/${type}/${id}.json",
|
|
TvType.TvSeries,
|
|
videos.map {
|
|
it.toEpisode(provider, type, imdbId)
|
|
}
|
|
) {
|
|
posterUrl = poster
|
|
backgroundPosterUrl = background
|
|
rating = imdbRating.toRatingInt()
|
|
plot = description
|
|
year = yearNum?.toIntOrNull()
|
|
tags = genre ?: genres
|
|
addActors(cast)
|
|
addTrailer(trailersSources?.map { "https://www.youtube.com/watch?v=${it.source}" }
|
|
?.randomOrNull())
|
|
addImdbId(imdbId)
|
|
}
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
private data class Video(
|
|
@JsonProperty("id") val id: String? = null,
|
|
@JsonProperty("title") val title: String? = null,
|
|
@JsonProperty("name") val name: String? = null,
|
|
@JsonProperty("season") val seasonNumber: Int? = null,
|
|
@JsonProperty("number") val number: Int? = null,
|
|
@JsonProperty("episode") val episode: Int? = null,
|
|
@JsonProperty("thumbnail") val thumbnail: String? = null,
|
|
@JsonProperty("overview") val overview: String? = null,
|
|
@JsonProperty("description") val description: String? = null,
|
|
) {
|
|
fun toEpisode(provider: StremioC, type: String?, imdbId: String?): Episode {
|
|
return provider.newEpisode(
|
|
LoadData(type, id, seasonNumber, episode ?: number, imdbId)
|
|
) {
|
|
this.name = name ?: title
|
|
this.posterUrl = thumbnail
|
|
this.description = overview ?: description
|
|
this.season = seasonNumber
|
|
this.episode = episode ?: number
|
|
}
|
|
}
|
|
}
|
|
|
|
private data class StreamsResponse(val streams: List<Stream>)
|
|
private data class Subtitle(
|
|
val url: String?,
|
|
val lang: String?,
|
|
val id: String?,
|
|
)
|
|
|
|
private data class ProxyHeaders(
|
|
val request: Map<String, String>?,
|
|
)
|
|
|
|
private data class BehaviorHints(
|
|
val proxyHeaders: ProxyHeaders?,
|
|
val headers: Map<String, String>?,
|
|
)
|
|
|
|
private data class Stream(
|
|
val name: String?,
|
|
val title: String?,
|
|
val url: String?,
|
|
val description: String?,
|
|
val ytId: String?,
|
|
val externalUrl: String?,
|
|
val behaviorHints: BehaviorHints?,
|
|
val infoHash: String?,
|
|
val sources: List<String> = emptyList(),
|
|
val subtitles: List<Subtitle> = emptyList()
|
|
) {
|
|
suspend fun runCallback(
|
|
subtitleCallback: (SubtitleFile) -> Unit,
|
|
callback: (ExtractorLink) -> Unit
|
|
) {
|
|
if (url != null) {
|
|
callback.invoke(
|
|
ExtractorLink(
|
|
name ?: "",
|
|
fixSourceName(name, title),
|
|
url,
|
|
"",
|
|
getQuality(listOf(description,title,name)),
|
|
headers = behaviorHints?.proxyHeaders?.request ?: behaviorHints?.headers
|
|
?: mapOf(),
|
|
type = INFER_TYPE
|
|
)
|
|
)
|
|
subtitles.map { sub ->
|
|
subtitleCallback.invoke(
|
|
SubtitleFile(
|
|
SubtitleHelper.fromThreeLettersToLanguage(sub.lang ?: "") ?: sub.lang
|
|
?: "",
|
|
sub.url ?: return@map
|
|
)
|
|
)
|
|
}
|
|
}
|
|
if (ytId != null) {
|
|
loadExtractor("https://www.youtube.com/watch?v=$ytId", subtitleCallback, callback)
|
|
}
|
|
if (externalUrl != null) {
|
|
loadExtractor(externalUrl, subtitleCallback, callback)
|
|
}
|
|
if (infoHash != null) {
|
|
val resp = app.get(TRACKER_LIST_URL).text
|
|
val otherTrackers = resp
|
|
.split("\n")
|
|
.filterIndexed { i, _ -> i % 2 == 0 }
|
|
.filter { s -> s.isNotEmpty() }.joinToString("") { "&tr=$it" }
|
|
|
|
val sourceTrackers = sources
|
|
.filter { it.startsWith("tracker:") }
|
|
.map { it.removePrefix("tracker:") }
|
|
.filter { s -> s.isNotEmpty() }.joinToString("") { "&tr=$it" }
|
|
|
|
val magnet = "magnet:?xt=urn:btih:${infoHash}${sourceTrackers}${otherTrackers}"
|
|
callback.invoke(
|
|
ExtractorLink(
|
|
name ?: "",
|
|
title ?: name ?: "",
|
|
magnet,
|
|
"",
|
|
Qualities.Unknown.value
|
|
)
|
|
)
|
|
}
|
|
}
|
|
}
|
|
}
|