forked from AlmightyHak/extensions-source
Initial commit
This commit is contained in:
commit
98ed7e8839
2263 changed files with 108711 additions and 0 deletions
15
src/en/animekhor/build.gradle
Normal file
15
src/en/animekhor/build.gradle
Normal file
|
@ -0,0 +1,15 @@
|
|||
ext {
|
||||
extName = 'AnimeKhor'
|
||||
extClass = '.AnimeKhor'
|
||||
themePkg = 'animestream'
|
||||
baseUrl = 'https://animekhor.xyz'
|
||||
overrideVersionCode = 3
|
||||
}
|
||||
|
||||
apply from: "$rootDir/common.gradle"
|
||||
|
||||
dependencies {
|
||||
implementation(project(':lib:okru-extractor'))
|
||||
implementation(project(':lib:streamwish-extractor'))
|
||||
implementation "dev.datlag.jsunpacker:jsunpacker:1.0.1"
|
||||
}
|
BIN
src/en/animekhor/res/mipmap-hdpi/ic_launcher.png
Normal file
BIN
src/en/animekhor/res/mipmap-hdpi/ic_launcher.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 3.7 KiB |
BIN
src/en/animekhor/res/mipmap-mdpi/ic_launcher.png
Normal file
BIN
src/en/animekhor/res/mipmap-mdpi/ic_launcher.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 2 KiB |
BIN
src/en/animekhor/res/mipmap-xhdpi/ic_launcher.png
Normal file
BIN
src/en/animekhor/res/mipmap-xhdpi/ic_launcher.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 5.4 KiB |
BIN
src/en/animekhor/res/mipmap-xxhdpi/ic_launcher.png
Normal file
BIN
src/en/animekhor/res/mipmap-xxhdpi/ic_launcher.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 10 KiB |
BIN
src/en/animekhor/res/mipmap-xxxhdpi/ic_launcher.png
Normal file
BIN
src/en/animekhor/res/mipmap-xxxhdpi/ic_launcher.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 15 KiB |
|
@ -0,0 +1,38 @@
|
|||
package eu.kanade.tachiyomi.animeextension.en.animekhor
|
||||
|
||||
import eu.kanade.tachiyomi.animeextension.en.animekhor.extractors.StreamHideExtractor
|
||||
import eu.kanade.tachiyomi.animesource.model.Video
|
||||
import eu.kanade.tachiyomi.lib.okruextractor.OkruExtractor
|
||||
import eu.kanade.tachiyomi.lib.streamwishextractor.StreamWishExtractor
|
||||
import eu.kanade.tachiyomi.multisrc.animestream.AnimeStream
|
||||
|
||||
class AnimeKhor : AnimeStream(
|
||||
"en",
|
||||
"AnimeKhor",
|
||||
"https://animekhor.xyz",
|
||||
) {
|
||||
// ============================ Video Links =============================
|
||||
|
||||
override fun getVideoList(url: String, name: String): List<Video> {
|
||||
val prefix = "$name - "
|
||||
return when {
|
||||
url.contains("ahvsh.com") || name.equals("streamhide", true) -> {
|
||||
StreamHideExtractor(client, headers).videosFromUrl(url, prefix = prefix)
|
||||
}
|
||||
url.contains("ok.ru") -> {
|
||||
OkruExtractor(client).videosFromUrl(url, prefix = prefix)
|
||||
}
|
||||
url.contains("streamwish") -> {
|
||||
val docHeaders = headers.newBuilder()
|
||||
.add("Referer", "$baseUrl/")
|
||||
.build()
|
||||
StreamWishExtractor(client, docHeaders).videosFromUrl(url, prefix)
|
||||
}
|
||||
// TODO: Videos won't play
|
||||
// url.contains("animeabc.xyz") -> {
|
||||
// AnimeABCExtractor(client, headers).videosFromUrl(url, prefix = prefix)
|
||||
// }
|
||||
else -> emptyList()
|
||||
}
|
||||
}
|
||||
}
|
|
@ -0,0 +1,52 @@
|
|||
package eu.kanade.tachiyomi.animeextension.en.animekhor.extractors
|
||||
|
||||
import eu.kanade.tachiyomi.animesource.model.Video
|
||||
import eu.kanade.tachiyomi.network.GET
|
||||
import eu.kanade.tachiyomi.util.asJsoup
|
||||
import kotlinx.serialization.Serializable
|
||||
import kotlinx.serialization.json.Json
|
||||
import okhttp3.Headers
|
||||
import okhttp3.HttpUrl.Companion.toHttpUrl
|
||||
import okhttp3.OkHttpClient
|
||||
import uy.kohesive.injekt.injectLazy
|
||||
|
||||
class AnimeABCExtractor(private val client: OkHttpClient, private val headers: Headers) {
|
||||
|
||||
private val json: Json by injectLazy()
|
||||
|
||||
fun videosFromUrl(url: String, prefix: String = ""): List<Video> {
|
||||
val videoList = mutableListOf<Video>()
|
||||
val document = client.newCall(GET(url, headers = headers)).execute().asJsoup()
|
||||
val data = document.selectFirst("script:containsData(m3u8)")?.data() ?: return emptyList()
|
||||
val sources = json.decodeFromString<List<Source>>(
|
||||
"[${data.substringAfter("sources:")
|
||||
.substringAfter("[")
|
||||
.substringBefore("]")}]",
|
||||
)
|
||||
sources.forEach { src ->
|
||||
val masterplHeaders = headers.newBuilder()
|
||||
.add("Accept", "*/*")
|
||||
.add("Connection", "keep-alive")
|
||||
.add("Host", url.toHttpUrl().host)
|
||||
.add("Referer", url)
|
||||
.build()
|
||||
val masterPlaylist = client.newCall(
|
||||
GET(src.file.replace("^//".toRegex(), "https://"), headers = masterplHeaders),
|
||||
).execute().body.string()
|
||||
|
||||
masterPlaylist.substringAfter("#EXT-X-STREAM-INF:").split("#EXT-X-STREAM-INF:").forEach {
|
||||
val quality = prefix + it.substringAfter("RESOLUTION=").substringAfter("x").substringBefore("\n") + "p ${src.label ?: ""}"
|
||||
val videoUrl = it.substringAfter("\n").substringBefore("\n")
|
||||
videoList.add(Video(videoUrl, quality, videoUrl, headers = masterplHeaders))
|
||||
}
|
||||
}
|
||||
|
||||
return videoList
|
||||
}
|
||||
|
||||
@Serializable
|
||||
data class Source(
|
||||
val file: String,
|
||||
val label: String? = null,
|
||||
)
|
||||
}
|
|
@ -0,0 +1,46 @@
|
|||
package eu.kanade.tachiyomi.animeextension.en.animekhor.extractors
|
||||
|
||||
import dev.datlag.jsunpacker.JsUnpacker
|
||||
import eu.kanade.tachiyomi.animesource.model.Track
|
||||
import eu.kanade.tachiyomi.animesource.model.Video
|
||||
import eu.kanade.tachiyomi.network.GET
|
||||
import okhttp3.Headers
|
||||
import okhttp3.OkHttpClient
|
||||
|
||||
class StreamHideExtractor(private val client: OkHttpClient, private val headers: Headers) {
|
||||
// from nineanime / ask4movie FilemoonExtractor
|
||||
private val subtitleRegex = Regex("""#EXT-X-MEDIA:TYPE=SUBTITLES.*?NAME="(.*?)".*?URI="(.*?)"""")
|
||||
|
||||
fun videosFromUrl(url: String, prefix: String = ""): List<Video> {
|
||||
val page = client.newCall(GET(url, headers = headers)).execute().body.string()
|
||||
val unpacked = JsUnpacker.unpackAndCombine(page) ?: page
|
||||
val playlistUrl = unpacked.substringAfter("sources:")
|
||||
.substringAfter("file:\"")
|
||||
.substringBefore('"')
|
||||
|
||||
val playlistData = client.newCall(GET(playlistUrl, headers = headers)).execute().body.string()
|
||||
|
||||
val subs = subtitleRegex.findAll(playlistData).map {
|
||||
val subUrl = fixUrl(it.groupValues[2], playlistUrl)
|
||||
Track(subUrl, it.groupValues[1])
|
||||
}.toList()
|
||||
|
||||
val separator = "#EXT-X-STREAM-INF"
|
||||
return playlistData.substringAfter(separator).split(separator).map {
|
||||
val resolution = it.substringAfter("RESOLUTION=")
|
||||
.substringBefore("\n")
|
||||
.substringAfter("x")
|
||||
.substringBefore(",") + "p"
|
||||
|
||||
val urlPart = it.substringAfter("\n").substringBefore("\n")
|
||||
val videoUrl = fixUrl(urlPart, playlistUrl)
|
||||
Video(videoUrl, "${prefix}StreamHide:$resolution", videoUrl, subtitleTracks = subs)
|
||||
}
|
||||
}
|
||||
|
||||
private fun fixUrl(urlPart: String, playlistUrl: String) =
|
||||
when {
|
||||
!urlPart.startsWith("https:") -> playlistUrl.substringBeforeLast("/") + "/$urlPart"
|
||||
else -> urlPart
|
||||
}
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue