mirror of
https://github.com/hexated/cloudstream-extensions-hexated.git
synced 2024-08-15 00:03:22 +00:00
added Moenime
This commit is contained in:
parent
2e5b5523ee
commit
8cbbcc6ae0
6 changed files with 294 additions and 0 deletions
2
.github/workflows/build.yml
vendored
2
.github/workflows/build.yml
vendored
|
@ -58,6 +58,7 @@ jobs:
|
||||||
PRIMEWIRE_KEY: ${{ secrets.PRIMEWIRE_KEY }}
|
PRIMEWIRE_KEY: ${{ secrets.PRIMEWIRE_KEY }}
|
||||||
ZSHOW_API: ${{ secrets.ZSHOW_API }}
|
ZSHOW_API: ${{ secrets.ZSHOW_API }}
|
||||||
SFMOVIES_API: ${{ secrets.SFMOVIES_API }}
|
SFMOVIES_API: ${{ secrets.SFMOVIES_API }}
|
||||||
|
MOENIME_API: ${{ secrets.MOENIME_API }}
|
||||||
run: |
|
run: |
|
||||||
cd $GITHUB_WORKSPACE/src
|
cd $GITHUB_WORKSPACE/src
|
||||||
echo SORA_API=$SORA_API >> local.properties
|
echo SORA_API=$SORA_API >> local.properties
|
||||||
|
@ -76,6 +77,7 @@ jobs:
|
||||||
echo PRIMEWIRE_KEY=$PRIMEWIRE_KEY >> local.properties
|
echo PRIMEWIRE_KEY=$PRIMEWIRE_KEY >> local.properties
|
||||||
echo ZSHOW_API=$ZSHOW_API >> local.properties
|
echo ZSHOW_API=$ZSHOW_API >> local.properties
|
||||||
echo SFMOVIES_API=$SFMOVIES_API >> local.properties
|
echo SFMOVIES_API=$SFMOVIES_API >> local.properties
|
||||||
|
echo MOENIME_API=$MOENIME_API >> local.properties
|
||||||
|
|
||||||
- name: Build Plugins
|
- name: Build Plugins
|
||||||
run: |
|
run: |
|
||||||
|
|
37
Moenime/build.gradle.kts
Normal file
37
Moenime/build.gradle.kts
Normal file
|
@ -0,0 +1,37 @@
|
||||||
|
import org.jetbrains.kotlin.konan.properties.Properties
|
||||||
|
|
||||||
|
// use an integer for version numbers
|
||||||
|
version = 1
|
||||||
|
|
||||||
|
android {
|
||||||
|
defaultConfig {
|
||||||
|
val properties = Properties()
|
||||||
|
properties.load(project.rootProject.file("local.properties").inputStream())
|
||||||
|
|
||||||
|
buildConfigField("String", "MOENIME_API", "\"${properties.getProperty("MOENIME_API")}\"")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
cloudstream {
|
||||||
|
language = "id"
|
||||||
|
// All of these properties are optional, you can safely remove them
|
||||||
|
|
||||||
|
// description = "Lorem Ipsum"
|
||||||
|
authors = listOf("Hexated")
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Status int as the following:
|
||||||
|
* 0: Down
|
||||||
|
* 1: Ok
|
||||||
|
* 2: Slow
|
||||||
|
* 3: Beta only
|
||||||
|
* */
|
||||||
|
status = 0 // will be 3 if unspecified
|
||||||
|
tvTypes = listOf(
|
||||||
|
"AnimeMovie",
|
||||||
|
"Anime",
|
||||||
|
"OVA",
|
||||||
|
)
|
||||||
|
|
||||||
|
iconUrl = "https://cdn.discordapp.com/attachments/1170001679085744209/1170001727332810802/fast-forward.png?ex=65577405&is=6544ff05&hm=bdc8c8a9325e31ead9d528fd44a142e2254f29961679eb5196981cf9c06d2171&"
|
||||||
|
}
|
2
Moenime/src/main/AndroidManifest.xml
Normal file
2
Moenime/src/main/AndroidManifest.xml
Normal file
|
@ -0,0 +1,2 @@
|
||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<manifest package="com.hexated"/>
|
14
Moenime/src/main/kotlin/com/hexated/Extractors.kt
Normal file
14
Moenime/src/main/kotlin/com/hexated/Extractors.kt
Normal file
|
@ -0,0 +1,14 @@
|
||||||
|
package com.hexated
|
||||||
|
|
||||||
|
import com.lagradost.cloudstream3.extractors.Filesim
|
||||||
|
import com.lagradost.cloudstream3.extractors.StreamSB
|
||||||
|
|
||||||
|
class Nyomo : StreamSB() {
|
||||||
|
override var name: String = "Nyomo"
|
||||||
|
override var mainUrl = "https://nyomo.my.id"
|
||||||
|
}
|
||||||
|
|
||||||
|
class Streamhide : Filesim() {
|
||||||
|
override var name: String = "Streamhide"
|
||||||
|
override var mainUrl: String = "https://streamhide.to"
|
||||||
|
}
|
223
Moenime/src/main/kotlin/com/hexated/Moenime.kt
Normal file
223
Moenime/src/main/kotlin/com/hexated/Moenime.kt
Normal file
|
@ -0,0 +1,223 @@
|
||||||
|
package com.hexated
|
||||||
|
|
||||||
|
import com.lagradost.cloudstream3.*
|
||||||
|
import com.lagradost.cloudstream3.LoadResponse.Companion.addAniListId
|
||||||
|
import com.lagradost.cloudstream3.LoadResponse.Companion.addMalId
|
||||||
|
import com.lagradost.cloudstream3.utils.ExtractorLink
|
||||||
|
import com.lagradost.cloudstream3.utils.Qualities
|
||||||
|
import com.lagradost.cloudstream3.utils.loadExtractor
|
||||||
|
import org.jsoup.Jsoup
|
||||||
|
import org.jsoup.nodes.Element
|
||||||
|
|
||||||
|
class Moenime : MainAPI() {
|
||||||
|
private var apiUrl = BuildConfig.MOENIME_API
|
||||||
|
override val instantLinkLoading = true
|
||||||
|
override var name = "Moenime"
|
||||||
|
override val hasMainPage = true
|
||||||
|
override var lang = "id"
|
||||||
|
private var headers: Map<String,String> = mapOf()
|
||||||
|
private var cookies: Map<String,String> = mapOf()
|
||||||
|
override val supportedTypes = setOf(
|
||||||
|
TvType.Anime,
|
||||||
|
TvType.AnimeMovie,
|
||||||
|
TvType.OVA
|
||||||
|
)
|
||||||
|
|
||||||
|
override val mainPage = mainPageOf(
|
||||||
|
"$apiUrl/anime/ongoing?order_by=updated&page=" to "Sedang Tayang",
|
||||||
|
"$apiUrl/anime/finished?order_by=updated&page=" to "Selesai Tayang",
|
||||||
|
"$apiUrl/properties/season/summer-2022?order_by=most_viewed&page=" to "Dilihat Terbanyak Musim Ini",
|
||||||
|
"$apiUrl/anime/movie?order_by=updated&page=" to "Film Layar Lebar",
|
||||||
|
)
|
||||||
|
|
||||||
|
override suspend fun getMainPage(
|
||||||
|
page: Int,
|
||||||
|
request: MainPageRequest
|
||||||
|
): HomePageResponse {
|
||||||
|
val document = app.get(request.data + page).document
|
||||||
|
|
||||||
|
val home = document.select("div.col-lg-4.col-md-6.col-sm-6").mapNotNull {
|
||||||
|
it.toSearchResult()
|
||||||
|
}
|
||||||
|
|
||||||
|
return newHomePageResponse(request.name, home)
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun getProperAnimeLink(uri: String): String {
|
||||||
|
return if (uri.contains("/episode")) {
|
||||||
|
Regex("(.*)/episode/.+").find(uri)?.groupValues?.get(1).toString() + "/"
|
||||||
|
} else {
|
||||||
|
uri
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun Element.toSearchResult(): AnimeSearchResponse? {
|
||||||
|
val href = getProperAnimeLink(fixUrl(this.selectFirst("a")!!.attr("href")))
|
||||||
|
val title = this.selectFirst("h5 a")?.text() ?: return null
|
||||||
|
val posterUrl = fixUrl(this.select("div.product__item__pic.set-bg").attr("data-setbg"))
|
||||||
|
val episode = this.select("div.ep span").text().let {
|
||||||
|
Regex("Ep\\s(\\d+)\\s/").find(it)?.groupValues?.getOrNull(1)?.toIntOrNull()
|
||||||
|
}
|
||||||
|
|
||||||
|
return newAnimeSearchResponse(title, href, TvType.Anime) {
|
||||||
|
this.posterUrl = posterUrl
|
||||||
|
addSub(episode)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
override suspend fun search(query: String): List<SearchResponse> {
|
||||||
|
val link = "$apiUrl/anime?search=$query&order_by=latest"
|
||||||
|
val document = app.get(link).document
|
||||||
|
|
||||||
|
return document.select("div#animeList div.product__item").mapNotNull {
|
||||||
|
it.toSearchResult()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
override suspend fun load(url: String): LoadResponse {
|
||||||
|
val document = app.get(url).document
|
||||||
|
|
||||||
|
val title = document.selectFirst(".anime__details__title > h3")!!.text().trim()
|
||||||
|
val poster = document.selectFirst(".anime__details__pic")?.attr("data-setbg")
|
||||||
|
val tags = document.select("div.anime__details__widget > div > div:nth-child(2) > ul > li:nth-child(1)")
|
||||||
|
.text().trim().replace("Genre: ", "").split(", ")
|
||||||
|
|
||||||
|
val year = Regex("\\D").replace(
|
||||||
|
document.select("div.anime__details__widget > div > div:nth-child(1) > ul > li:nth-child(5)")
|
||||||
|
.text().trim().replace("Musim: ", ""), ""
|
||||||
|
).toIntOrNull()
|
||||||
|
val status = getStatus(
|
||||||
|
document.select("div.anime__details__widget > div > div:nth-child(1) > ul > li:nth-child(3)")
|
||||||
|
.text().trim().replace("Status: ", "")
|
||||||
|
)
|
||||||
|
val description = document.select(".anime__details__text > p").text().trim()
|
||||||
|
|
||||||
|
val episodes = mutableListOf<Episode>()
|
||||||
|
|
||||||
|
for (i in 1..6) {
|
||||||
|
val doc = app.get("$url?page=$i").document
|
||||||
|
val eps = Jsoup.parse(doc.select("#episodeLists").attr("data-content")).select("a.btn.btn-sm.btn-danger")
|
||||||
|
.mapNotNull {
|
||||||
|
val name = it.text().trim()
|
||||||
|
val episode = Regex("(\\d+[.,]?\\d*)").find(name)?.groupValues?.getOrNull(0)
|
||||||
|
?.toIntOrNull()
|
||||||
|
val link = it.attr("href")
|
||||||
|
Episode(link, episode = episode)
|
||||||
|
}
|
||||||
|
if(eps.isEmpty()) break else episodes.addAll(eps)
|
||||||
|
}
|
||||||
|
|
||||||
|
val type = getType(document.selectFirst("div.col-lg-6.col-md-6 ul li:contains(Tipe:) a")?.text()?.lowercase() ?: "tv", episodes.size)
|
||||||
|
val recommendations = document.select("div#randomList > a").mapNotNull {
|
||||||
|
val epHref = it.attr("href")
|
||||||
|
val epTitle = it.select("h5.sidebar-title-h5.px-2.py-2").text()
|
||||||
|
val epPoster = it.select(".product__sidebar__view__item.set-bg").attr("data-setbg")
|
||||||
|
newAnimeSearchResponse(epTitle, epHref, TvType.Anime) {
|
||||||
|
this.posterUrl = epPoster
|
||||||
|
addDubStatus(dubExist = false, subExist = true)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
val tracker = APIHolder.getTracker(listOf(title),TrackerType.getTypes(type),year,true)
|
||||||
|
|
||||||
|
return newAnimeLoadResponse(title, url, type) {
|
||||||
|
engName = title
|
||||||
|
posterUrl = tracker?.image ?: poster
|
||||||
|
backgroundPosterUrl = tracker?.cover
|
||||||
|
this.year = year
|
||||||
|
addEpisodes(DubStatus.Subbed, episodes)
|
||||||
|
showStatus = status
|
||||||
|
plot = description
|
||||||
|
this.tags = tags
|
||||||
|
this.recommendations = recommendations
|
||||||
|
addMalId(tracker?.malId)
|
||||||
|
addAniListId(tracker?.aniId?.toIntOrNull())
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
private suspend fun invokeLocalSource(
|
||||||
|
url: String,
|
||||||
|
server: String,
|
||||||
|
ref: String,
|
||||||
|
callback: (ExtractorLink) -> Unit
|
||||||
|
) {
|
||||||
|
val name = if(server.contains("drive")) "Main Server" else "Backup Server"
|
||||||
|
val document = app.get(
|
||||||
|
url,
|
||||||
|
referer = ref,
|
||||||
|
headers = headers,
|
||||||
|
cookies = cookies
|
||||||
|
).document
|
||||||
|
document.select("video#player > source").map {
|
||||||
|
val link = fixUrl(it.attr("src"))
|
||||||
|
val quality = it.attr("size").toIntOrNull()
|
||||||
|
callback.invoke(
|
||||||
|
ExtractorLink(
|
||||||
|
name,
|
||||||
|
name,
|
||||||
|
link,
|
||||||
|
referer = "",
|
||||||
|
quality = quality ?: Qualities.Unknown.value,
|
||||||
|
)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
override suspend fun loadLinks(
|
||||||
|
data: String,
|
||||||
|
isCasting: Boolean,
|
||||||
|
subtitleCallback: (SubtitleFile) -> Unit,
|
||||||
|
callback: (ExtractorLink) -> Unit
|
||||||
|
): Boolean {
|
||||||
|
val req = app.get(data)
|
||||||
|
val res = req.document
|
||||||
|
val token = res.select("meta[name=csrf-token]").attr("content")
|
||||||
|
headers = mapOf(
|
||||||
|
"X-Requested-With" to "XMLHttpRequest",
|
||||||
|
"X-CSRF-TOKEN" to token
|
||||||
|
)
|
||||||
|
cookies = req.cookies.toMutableMap()
|
||||||
|
.plus(mapOf(sessions to value))
|
||||||
|
res.select("select#changeServer option").apmap { source ->
|
||||||
|
val server = source.attr("value")
|
||||||
|
val link = "$data?dfgRr1OagZvvxbzHNpyCy0FqJQ18mCnb=XNvyMgJO6J&twEvZlbZbYRWBdKKwxkOnwYF0VWoGGVg=$server"
|
||||||
|
if (server.contains(Regex("(?i)$server1|$server2"))) {
|
||||||
|
invokeLocalSource(link, server, data, callback)
|
||||||
|
} else {
|
||||||
|
app.get(
|
||||||
|
link,
|
||||||
|
referer = data,
|
||||||
|
headers = headers,
|
||||||
|
cookies = cookies
|
||||||
|
).document.select("div.iframe-container iframe").attr("src").let { videoUrl ->
|
||||||
|
loadExtractor(fixUrl(videoUrl), "$apiUrl/", subtitleCallback, callback)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
companion object {
|
||||||
|
private var sessions = base64Decode("a3VyYW1hbmltZV9zZXNzaW9ucw==")
|
||||||
|
private var value = base64Decode("VXV3ZENIR1B3NVVBYlBTdDRpYXpUZFNpUHpCZXBhd2pEMmJoWjFDYWRkUzYyUUUwMlRLZVZtYWpKNnFKeWJ3SA==")
|
||||||
|
private val server1 = base64Decode("a3VyYW1hZHJpdmU=")
|
||||||
|
private val server2 = base64Decode("YXJjaGl2ZQ==")
|
||||||
|
|
||||||
|
fun getType(t: String, s: Int): TvType {
|
||||||
|
return if (t.contains("OVA", true) || t.contains("Special")) TvType.OVA
|
||||||
|
else if (t.contains("Movie", true) && s == 1) TvType.AnimeMovie
|
||||||
|
else TvType.Anime
|
||||||
|
}
|
||||||
|
|
||||||
|
fun getStatus(t: String): ShowStatus {
|
||||||
|
return when (t) {
|
||||||
|
"Selesai Tayang" -> ShowStatus.Completed
|
||||||
|
"Sedang Tayang" -> ShowStatus.Ongoing
|
||||||
|
else -> ShowStatus.Completed
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
16
Moenime/src/main/kotlin/com/hexated/MoenimePlugin.kt
Normal file
16
Moenime/src/main/kotlin/com/hexated/MoenimePlugin.kt
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
|
||||||
|
package com.hexated
|
||||||
|
|
||||||
|
import com.lagradost.cloudstream3.plugins.CloudstreamPlugin
|
||||||
|
import com.lagradost.cloudstream3.plugins.Plugin
|
||||||
|
import android.content.Context
|
||||||
|
|
||||||
|
@CloudstreamPlugin
|
||||||
|
class MoenimePlugin: Plugin() {
|
||||||
|
override fun load(context: Context) {
|
||||||
|
// All providers should be added in this manner. Please don't edit the providers list directly.
|
||||||
|
registerMainAPI(Moenime())
|
||||||
|
registerExtractorAPI(Nyomo())
|
||||||
|
registerExtractorAPI(Streamhide())
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in a new issue