sora: added aniwave & fix few sources

This commit is contained in:
alex 2024-01-12 14:17:54 +07:00
parent b77d6ead50
commit 081e5aa1fd
7 changed files with 394 additions and 252 deletions

View file

@ -1,7 +1,7 @@
import org.jetbrains.kotlin.konan.properties.Properties import org.jetbrains.kotlin.konan.properties.Properties
// use an integer for version numbers // use an integer for version numbers
version = 215 version = 216
android { android {
defaultConfig { defaultConfig {

View file

@ -263,8 +263,7 @@ open class Streamruby : ExtractorApi() {
} else { } else {
response.document.selectFirst("script:containsData(sources:)")?.data() response.document.selectFirst("script:containsData(sources:)")?.data()
} }
val m3u8 = val m3u8 = Regex("file:\\s*\"(.*?m3u8.*?)\"").find(script ?: return)?.groupValues?.getOrNull(1)
Regex("file:\\s*\"(.*?m3u8.*?)\"").find(script ?: return)?.groupValues?.getOrNull(1)
M3u8Helper.generateM3u8( M3u8Helper.generateM3u8(
name, name,
m3u8 ?: return, m3u8 ?: return,
@ -466,7 +465,7 @@ class Animefever : Filesim() {
override var mainUrl = "https://animefever.fun" override var mainUrl = "https://animefever.fun"
} }
class Multimovies : Filesim() { class Multimovies : Ridoo() {
override val name = "Multimovies" override val name = "Multimovies"
override var mainUrl = "https://multimovies.cloud" override var mainUrl = "https://multimovies.cloud"
} }

View file

@ -492,7 +492,8 @@ object SoraExtractor : SoraStream() {
val req = app.get(source, referer = "$host/") val req = app.get(source, referer = "$host/")
val server = getBaseUrl(req.url) val server = getBaseUrl(req.url)
val script = req.text.substringAfter("wc0 = '").substringBefore("'") val script = req.text.substringAfter("wc0 = '").substringBefore("'")
val video = tryParseJson<Map<String, String>>(base64Decode(script))?.get("file") val video =
tryParseJson<Map<String, String>>(base64Decode(script))?.get("file")
M3u8Helper.generateM3u8( M3u8Helper.generateM3u8(
"Voe", "Voe",
video ?: return@apmap, video ?: return@apmap,
@ -781,9 +782,11 @@ object SoraExtractor : SoraStream() {
val mediaId = app.get(url).document.selectFirst("ul.episodes li a")?.attr("data-id") val mediaId = app.get(url).document.selectFirst("ul.episodes li a")?.attr("data-id")
?: return ?: return
app.get("$vidsrctoAPI/ajax/embed/episode/$mediaId/sources", headers = mapOf( app.get(
"$vidsrctoAPI/ajax/embed/episode/$mediaId/sources", headers = mapOf(
"X-Requested-With" to "XMLHttpRequest" "X-Requested-With" to "XMLHttpRequest"
)).parsedSafe<VidsrctoSources>()?.result?.apmap { )
).parsedSafe<VidsrctoSources>()?.result?.apmap {
val encUrl = app.get("$vidsrctoAPI/ajax/embed/source/${it.id}") val encUrl = app.get("$vidsrctoAPI/ajax/embed/source/${it.id}")
.parsedSafe<VidsrctoResponse>()?.result?.url .parsedSafe<VidsrctoResponse>()?.result?.url
loadExtractor( loadExtractor(
@ -905,11 +908,23 @@ object SoraExtractor : SoraStream() {
if (season == null) TvType.AnimeMovie else TvType.Anime if (season == null) TvType.AnimeMovie else TvType.Anime
) )
argamap({ val malsync = app.get("$malsyncAPI/mal/anime/${malId ?: return}")
.parsedSafe<MALSyncResponses>()?.sites
val zoroIds = malsync?.zoro?.keys?.map { it }
val aniwaveId = malsync?.nineAnime?.firstNotNullOf { it.value["url"] }
argamap(
{
invokeAnimetosho(malId, season, episode, subtitleCallback, callback) invokeAnimetosho(malId, season, episode, subtitleCallback, callback)
}, { },
invokeAniwatch(malId, episode, subtitleCallback, callback) {
}, { invokeAniwatch(zoroIds, episode, subtitleCallback, callback)
},
{
invokeAniwave(aniwaveId, episode, subtitleCallback, callback)
},
{
if (season != null) invokeCrunchyroll( if (season != null) invokeCrunchyroll(
aniId, aniId,
malId, malId,
@ -919,7 +934,42 @@ object SoraExtractor : SoraStream() {
subtitleCallback, subtitleCallback,
callback callback
) )
}) }
)
}
private suspend fun invokeAniwave(
url: String? = null,
episode: Int? = null,
subtitleCallback: (SubtitleFile) -> Unit,
callback: (ExtractorLink) -> Unit
) {
val res = app.get(url ?: return).document
val id = res.select("div#watch-main").attr("data-id")
val episodeId =
app.get("$aniwaveAPI/ajax/episode/list/$id?vrf=${AniwaveUtils.encodeVrf(id)}")
.parsedSafe<AniwaveResponse>()?.asJsoup()
?.selectFirst("ul.ep-range li a[data-num=${episode ?: 1}]")?.attr("data-ids")
?: return
val servers =
app.get("$aniwaveAPI/ajax/server/list/$episodeId?vrf=${AniwaveUtils.encodeVrf(episodeId)}")
.parsedSafe<AniwaveResponse>()?.asJsoup()
?.select("div.servers > div[data-type!=sub] ul li") ?: return
servers.apmap {
val linkId = it.attr("data-link-id")
val iframe =
app.get("$aniwaveAPI/ajax/server/$linkId?vrf=${AniwaveUtils.encodeVrf(linkId)}")
.parsedSafe<AniwaveServer>()?.result?.decrypt()
val audio = if (it.attr("data-cmid").endsWith("softsub")) "Raw" else "English Dub"
loadCustomExtractor(
"${it.text()} [$audio]",
iframe ?: return@apmap,
"$aniwaveAPI/",
subtitleCallback,
callback,
)
}
} }
private suspend fun invokeAnimetosho( private suspend fun invokeAnimetosho(
@ -968,7 +1018,7 @@ object SoraExtractor : SoraStream() {
} }
private suspend fun invokeAniwatch( private suspend fun invokeAniwatch(
malId: Int? = null, animeIds: List<String?>? = null,
episode: Int? = null, episode: Int? = null,
subtitleCallback: (SubtitleFile) -> Unit, subtitleCallback: (SubtitleFile) -> Unit,
callback: (ExtractorLink) -> Unit callback: (ExtractorLink) -> Unit
@ -976,9 +1026,7 @@ object SoraExtractor : SoraStream() {
val headers = mapOf( val headers = mapOf(
"X-Requested-With" to "XMLHttpRequest", "X-Requested-With" to "XMLHttpRequest",
) )
val animeId = app.get("$malsyncAPI/mal/anime/${malId ?: return}") animeIds?.apmap { id ->
.parsedSafe<MALSyncResponses>()?.sites?.zoro?.keys?.map { it }
animeId?.apmap { id ->
val episodeId = app.get( val episodeId = app.get(
"$aniwatchAPI/ajax/v2/episode/list/${id ?: return@apmap}", "$aniwatchAPI/ajax/v2/episode/list/${id ?: return@apmap}",
headers = headers headers = headers
@ -2372,14 +2420,14 @@ object SoraExtractor : SoraStream() {
"""postid\":\"""" """postid\":\""""
).substringBefore("""\"""") ).substringBefore("""\"""")
} ?: return } ?: return
val url = if(season == null) { val url = if (season == null) {
"$ridomoviesAPI/core/api/movies/$slug/videos" "$ridomoviesAPI/core/api/movies/$slug/videos"
} else { } else {
"$ridomoviesAPI/core/api/episodes/$slug/videos" "$ridomoviesAPI/core/api/episodes/$slug/videos"
} }
app.get(url).parsedSafe<RidoResponses>()?.data?.apmap { link -> app.get(url).parsedSafe<RidoResponses>()?.data?.apmap { link ->
val iframe = Jsoup.parse(link.url ?: return@apmap).select("iframe").attr("data-src") val iframe = Jsoup.parse(link.url ?: return@apmap).select("iframe").attr("data-src")
if(iframe.startsWith("https://closeload.top")) { if (iframe.startsWith("https://closeload.top")) {
val unpacked = val unpacked =
getAndUnpack( getAndUnpack(
app.get( app.get(

View file

@ -1,6 +1,8 @@
package com.hexated package com.hexated
import com.fasterxml.jackson.annotation.JsonProperty import com.fasterxml.jackson.annotation.JsonProperty
import org.jsoup.Jsoup
import org.jsoup.nodes.Document
data class CrunchyrollAccessToken( data class CrunchyrollAccessToken(
val accessToken: String? = null, val accessToken: String? = null,
@ -25,7 +27,30 @@ data class TmdbDate(
val nextWeek: String, val nextWeek: String,
) )
data class AniMedia(@JsonProperty("id") var id: Int? = null, @JsonProperty("idMal") var idMal: Int? = null) data class AniwaveResponse(
val result: String
) {
fun asJsoup(): Document {
return Jsoup.parse(result)
}
}
data class AniwaveServer(
val result: Result
) {
data class Result(
val url: String
) {
fun decrypt(): String {
return AniwaveUtils.decodeVrf(url)
}
}
}
data class AniMedia(
@JsonProperty("id") var id: Int? = null,
@JsonProperty("idMal") var idMal: Int? = null
)
data class AniPage(@JsonProperty("media") var media: java.util.ArrayList<AniMedia> = arrayListOf()) data class AniPage(@JsonProperty("media") var media: java.util.ArrayList<AniMedia> = arrayListOf())
@ -235,6 +260,7 @@ data class CrunchyrollSourcesResponses(
data class MALSyncSites( data class MALSyncSites(
@JsonProperty("Zoro") val zoro: HashMap<String?, HashMap<String, String?>>? = hashMapOf(), @JsonProperty("Zoro") val zoro: HashMap<String?, HashMap<String, String?>>? = hashMapOf(),
@JsonProperty("9anime") val nineAnime: HashMap<String?, HashMap<String, String?>>? = hashMapOf(),
) )
data class MALSyncResponses( data class MALSyncResponses(
@ -364,7 +390,10 @@ data class SFMoviesAttributes(
@JsonProperty("contentId") var contentId: String? = null, @JsonProperty("contentId") var contentId: String? = null,
) )
data class SFMoviesData(@JsonProperty("id") var id: Int? = null, @JsonProperty("attributes") var attributes: SFMoviesAttributes? = SFMoviesAttributes()) data class SFMoviesData(
@JsonProperty("id") var id: Int? = null,
@JsonProperty("attributes") var attributes: SFMoviesAttributes? = SFMoviesAttributes()
)
data class SFMoviesSearch( data class SFMoviesSearch(
@JsonProperty("data") var data: ArrayList<SFMoviesData>? = arrayListOf(), @JsonProperty("data") var data: ArrayList<SFMoviesData>? = arrayListOf(),

View file

@ -91,6 +91,7 @@ open class SoraStream : TmdbProvider() {
const val filmxyAPI = "https://www.filmxy.vip" const val filmxyAPI = "https://www.filmxy.vip"
const val kimcartoonAPI = "https://kimcartoon.li" const val kimcartoonAPI = "https://kimcartoon.li"
const val aniwatchAPI = "https://aniwatch.to" const val aniwatchAPI = "https://aniwatch.to"
const val aniwaveAPI = "https://aniwave.to"
const val crunchyrollAPI = "https://beta-api.crunchyroll.com" const val crunchyrollAPI = "https://beta-api.crunchyroll.com"
const val kissKhAPI = "https://kisskh.co" const val kissKhAPI = "https://kisskh.co"
const val lingAPI = "https://ling-online.net" const val lingAPI = "https://ling-online.net"
@ -584,7 +585,15 @@ open class SoraStream : TmdbProvider() {
if (!res.isAnime) invokeNowTv(res.id, res.imdbId, res.season, res.episode, callback) if (!res.isAnime) invokeNowTv(res.id, res.imdbId, res.season, res.episode, callback)
}, },
{ {
if (!res.isAnime && res.season == null) invokeRidomovies(res.id, res.imdbId, res.title, res.season, res.episode, subtitleCallback, callback) if (!res.isAnime) invokeRidomovies(
res.id,
res.imdbId,
res.title,
res.season,
res.episode,
subtitleCallback,
callback
)
}, },
{ {
if (!res.isAnime) invokeAllMovieland(res.imdbId, res.season, res.episode, callback) if (!res.isAnime) invokeAllMovieland(res.imdbId, res.season, res.episode, callback)

View file

@ -223,7 +223,15 @@ class SoraStreamLite : SoraStream() {
) )
}, },
{ {
if (!res.isAnime && res.season == null) invokeRidomovies(res.id, res.imdbId, res.title, res.season, res.episode, subtitleCallback, callback) if (!res.isAnime) invokeRidomovies(
res.id,
res.imdbId,
res.title,
res.season,
res.episode,
subtitleCallback,
callback
)
}, },
{ {
if (!res.isAnime) invokeEmovies( if (!res.isAnime) invokeEmovies(

View file

@ -1304,6 +1304,55 @@ suspend fun request(
return client.newCall(request).await() return client.newCall(request).await()
} }
// steal from https://github.com/aniyomiorg/aniyomi-extensions/blob/master/src/en/aniwave/src/eu/kanade/tachiyomi/animeextension/en/nineanime/AniwaveUtils.kt
// credits to @samfundev
object AniwaveUtils {
fun encodeVrf(input: String): String {
val rc4Key = SecretKeySpec("ysJhV6U27FVIjjuk".toByteArray(), "RC4")
val cipher = Cipher.getInstance("RC4")
cipher.init(Cipher.DECRYPT_MODE, rc4Key, cipher.parameters)
var vrf = cipher.doFinal(input.toByteArray())
vrf = Base64.encode(vrf, Base64.URL_SAFE or Base64.NO_WRAP)
vrf = Base64.encode(vrf, Base64.DEFAULT or Base64.NO_WRAP)
vrf = vrfShift(vrf)
vrf = Base64.encode(vrf, Base64.DEFAULT)
vrf = rot13(vrf)
val stringVrf = vrf.toString(Charsets.UTF_8)
return encode(stringVrf)
}
fun decodeVrf(input: String): String {
var vrf = input.toByteArray()
vrf = Base64.decode(vrf, Base64.URL_SAFE)
val rc4Key = SecretKeySpec("hlPeNwkncH0fq9so".toByteArray(), "RC4")
val cipher = Cipher.getInstance("RC4")
cipher.init(Cipher.DECRYPT_MODE, rc4Key, cipher.parameters)
vrf = cipher.doFinal(vrf)
return decode(vrf.toString(Charsets.UTF_8))
}
private fun rot13(vrf: ByteArray): ByteArray {
for (i in vrf.indices) {
val byte = vrf[i]
if (byte in 'A'.code..'Z'.code) {
vrf[i] = ((byte - 'A'.code + 13) % 26 + 'A'.code).toByte()
} else if (byte in 'a'.code..'z'.code) {
vrf[i] = ((byte - 'a'.code + 13) % 26 + 'a'.code).toByte()
}
}
return vrf
}
private fun vrfShift(vrf: ByteArray): ByteArray {
for (i in vrf.indices) {
val shift = arrayOf(-3, 3, -4, 2, -2, 5, 4, 5)[i % 8]
vrf[i] = vrf[i].plus(shift).toByte()
}
return vrf
}
}
object DumpUtils { object DumpUtils {
private val deviceId = getDeviceId() private val deviceId = getDeviceId()