2023-04-01 17:42:02 +00:00
|
|
|
package com.hexated
|
|
|
|
|
|
|
|
import android.util.Log
|
|
|
|
import com.lagradost.cloudstream3.*
|
|
|
|
import com.lagradost.cloudstream3.utils.*
|
|
|
|
import com.lagradost.cloudstream3.utils.AppUtils.parseJson
|
|
|
|
import com.lagradost.cloudstream3.utils.AppUtils.toJson
|
|
|
|
import com.lagradost.cloudstream3.utils.AppUtils.tryParseJson
|
|
|
|
import org.json.JSONObject
|
|
|
|
import java.net.URI
|
|
|
|
|
|
|
|
private const val TRACKER_LIST_URL =
|
|
|
|
"https://raw.githubusercontent.com/ngosang/trackerslist/master/trackers_best.txt"
|
|
|
|
|
|
|
|
class Stremio : MainAPI() {
|
|
|
|
override var mainUrl = "https://stremio.github.io/stremio-static-addon-example"
|
|
|
|
override var name = "Stremio"
|
|
|
|
override val supportedTypes = setOf(TvType.Others)
|
|
|
|
override val hasMainPage = true
|
2023-04-01 20:36:49 +00:00
|
|
|
private var fixedUrl = mainUrl
|
2023-04-01 17:42:02 +00:00
|
|
|
|
|
|
|
override suspend fun getMainPage(page: Int, request: MainPageRequest): HomePageResponse? {
|
2023-04-01 20:36:49 +00:00
|
|
|
fixedUrl = mainUrl.fixSourceUrl()
|
2023-04-01 18:27:43 +00:00
|
|
|
val res = tryParseJson<Manifest>(app.get("${fixedUrl}/manifest.json").text) ?: return null
|
2023-04-01 17:42:02 +00:00
|
|
|
val lists = mutableListOf<HomePageList>()
|
|
|
|
res.catalogs.forEach { catalog ->
|
|
|
|
catalog.toHomePageList(this)?.let {
|
|
|
|
lists.add(it)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return HomePageResponse(
|
|
|
|
lists,
|
|
|
|
false
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
override suspend fun search(query: String): List<SearchResponse>? {
|
2023-04-01 20:36:49 +00:00
|
|
|
fixedUrl = mainUrl.fixSourceUrl()
|
2023-04-01 18:27:43 +00:00
|
|
|
val res = tryParseJson<Manifest>(app.get("${fixedUrl}/manifest.json").text) ?: return null
|
2023-04-01 17:42:02 +00:00
|
|
|
val list = mutableListOf<SearchResponse>()
|
|
|
|
res.catalogs.forEach { catalog ->
|
|
|
|
list.addAll(catalog.search(query, this))
|
|
|
|
}
|
|
|
|
return list
|
|
|
|
}
|
|
|
|
|
|
|
|
override suspend fun load(url: String): LoadResponse? {
|
|
|
|
val res = parseJson<CatalogEntry>(url)
|
2023-04-01 18:27:43 +00:00
|
|
|
val json = app.get("${fixedUrl}/meta/${res.type}/${res.id}.json")
|
2023-04-01 17:42:02 +00:00
|
|
|
.parsedSafe<CatalogResponse>()?.meta ?: throw RuntimeException(url)
|
|
|
|
return json.toLoadResponse(this)
|
|
|
|
}
|
|
|
|
|
|
|
|
override suspend fun loadLinks(
|
|
|
|
data: String,
|
|
|
|
isCasting: Boolean,
|
|
|
|
subtitleCallback: (SubtitleFile) -> Unit,
|
|
|
|
callback: (ExtractorLink) -> Unit
|
|
|
|
): Boolean {
|
|
|
|
val res = tryParseJson<StreamsResponse>(app.get(data).text) ?: return false
|
|
|
|
res.streams.forEach { stream ->
|
|
|
|
stream.runCallback(this, subtitleCallback, callback)
|
|
|
|
}
|
|
|
|
|
|
|
|
return 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: Stremio): List<SearchResponse> {
|
|
|
|
val entries = mutableListOf<SearchResponse>()
|
|
|
|
types.forEach { type ->
|
|
|
|
val json =
|
2023-04-01 18:27:43 +00:00
|
|
|
app.get("${provider.fixedUrl}/catalog/${type}/${id}/search=${query}.json").text
|
2023-04-01 17:42:02 +00:00
|
|
|
val res =
|
|
|
|
tryParseJson<CatalogResponse>(json)
|
|
|
|
?: return@forEach
|
|
|
|
res.metas?.forEach { entry ->
|
|
|
|
entries.add(entry.toSearchResponse(provider))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return entries
|
|
|
|
}
|
|
|
|
|
|
|
|
suspend fun toHomePageList(provider: Stremio): HomePageList? {
|
|
|
|
val entries = mutableListOf<SearchResponse>()
|
|
|
|
types.forEach { type ->
|
2023-04-01 18:27:43 +00:00
|
|
|
val json = app.get("${provider.fixedUrl}/catalog/${type}/${id}.json").text
|
2023-04-01 17:42:02 +00:00
|
|
|
val res =
|
|
|
|
tryParseJson<CatalogResponse>(json)
|
|
|
|
?: return@forEach
|
|
|
|
res.metas?.forEach { entry ->
|
|
|
|
entries.add(entry.toSearchResponse(provider))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return HomePageList(
|
|
|
|
name ?: id,
|
|
|
|
entries
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private data class CatalogResponse(val metas: List<CatalogEntry>?, val meta: CatalogEntry?)
|
|
|
|
private data class CatalogEntry(
|
|
|
|
val name: String,
|
|
|
|
val id: String,
|
|
|
|
val poster: String?,
|
|
|
|
val description: String?,
|
|
|
|
val type: String?,
|
|
|
|
val videos: List<Video>?
|
|
|
|
) {
|
|
|
|
fun toSearchResponse(provider: Stremio): SearchResponse {
|
|
|
|
return provider.newMovieSearchResponse(
|
|
|
|
fixTitle(name),
|
|
|
|
this.toJson(),
|
|
|
|
TvType.Others
|
|
|
|
) {
|
|
|
|
posterUrl = poster
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
suspend fun toLoadResponse(provider: Stremio): LoadResponse {
|
|
|
|
if (videos == null || videos.isEmpty()) {
|
|
|
|
return provider.newMovieLoadResponse(
|
|
|
|
name,
|
2023-04-01 18:27:43 +00:00
|
|
|
"${provider.fixedUrl}/meta/${type}/${id}.json",
|
2023-04-01 17:42:02 +00:00
|
|
|
TvType.Others,
|
2023-04-01 18:27:43 +00:00
|
|
|
"${provider.fixedUrl}/stream/${type}/${id}.json"
|
2023-04-01 17:42:02 +00:00
|
|
|
) {
|
|
|
|
posterUrl = poster
|
|
|
|
plot = description
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return provider.newTvSeriesLoadResponse(
|
|
|
|
name,
|
2023-04-01 18:27:43 +00:00
|
|
|
"${provider.fixedUrl}/meta/${type}/${id}.json",
|
2023-04-01 17:42:02 +00:00
|
|
|
TvType.Others,
|
|
|
|
videos.map {
|
|
|
|
it.toEpisode(provider, type)
|
|
|
|
}
|
|
|
|
) {
|
|
|
|
posterUrl = poster
|
|
|
|
plot = description
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private data class Video(
|
|
|
|
val id: String,
|
|
|
|
val title: String?,
|
|
|
|
val thumbnail: String?,
|
|
|
|
val overview: String?
|
|
|
|
) {
|
|
|
|
fun toEpisode(provider: Stremio, type: String?): Episode {
|
|
|
|
return provider.newEpisode(
|
2023-04-01 18:27:43 +00:00
|
|
|
"${provider.fixedUrl}/stream/${type}/${id}.json"
|
2023-04-01 17:42:02 +00:00
|
|
|
) {
|
|
|
|
this.name = title
|
|
|
|
this.posterUrl = thumbnail
|
|
|
|
this.description = overview
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private data class StreamsResponse(val streams: List<Stream>)
|
|
|
|
private data class Subtitle(
|
|
|
|
val url: String?,
|
|
|
|
val lang: String?,
|
|
|
|
val id: 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: JSONObject?,
|
|
|
|
val infoHash: String?,
|
|
|
|
val sources: List<String> = emptyList(),
|
|
|
|
val subtitles: List<Subtitle> = emptyList()
|
|
|
|
) {
|
|
|
|
suspend fun runCallback(
|
|
|
|
provider: Stremio,
|
|
|
|
subtitleCallback: (SubtitleFile) -> Unit,
|
|
|
|
callback: (ExtractorLink) -> Unit
|
|
|
|
) {
|
|
|
|
if (url != null) {
|
|
|
|
var referer: String? = null
|
|
|
|
try {
|
|
|
|
val headers = ((behaviorHints?.get("proxyHeaders") as? JSONObject)
|
|
|
|
?.get("request") as? JSONObject)
|
|
|
|
referer =
|
|
|
|
headers?.get("referer") as? String ?: headers?.get("origin") as? String
|
|
|
|
} catch (ex: Throwable) {
|
|
|
|
Log.e("Stremio", Log.getStackTraceString(ex))
|
|
|
|
}
|
|
|
|
callback.invoke(
|
|
|
|
ExtractorLink(
|
|
|
|
name ?: "",
|
|
|
|
title ?: name ?: "",
|
|
|
|
url,
|
2023-04-01 18:27:43 +00:00
|
|
|
if (provider.fixedUrl.contains("kisskh")) "https://kisskh.me/" else referer
|
2023-04-01 17:42:02 +00:00
|
|
|
?: "",
|
|
|
|
getQualityFromName(description),
|
|
|
|
isM3u8 = URI(url).path.endsWith(".m3u8")
|
|
|
|
)
|
|
|
|
)
|
|
|
|
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, s -> 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
|
|
|
|
)
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|