2022-12-02 13:00:44 +00:00
|
|
|
|
package com.hexated
|
|
|
|
|
|
2022-12-07 19:17:24 +00:00
|
|
|
|
import com.hexated.SoraStream.Companion.filmxyAPI
|
2022-12-02 16:50:56 +00:00
|
|
|
|
import com.hexated.SoraStream.Companion.gdbot
|
|
|
|
|
import com.lagradost.cloudstream3.app
|
2022-12-02 13:00:44 +00:00
|
|
|
|
import com.lagradost.cloudstream3.base64Decode
|
|
|
|
|
import com.lagradost.cloudstream3.base64Encode
|
|
|
|
|
import com.lagradost.cloudstream3.network.WebViewResolver
|
2022-12-07 15:06:48 +00:00
|
|
|
|
import com.lagradost.cloudstream3.utils.AppUtils
|
2022-12-07 19:17:24 +00:00
|
|
|
|
import com.lagradost.cloudstream3.utils.AppUtils.tryParseJson
|
2022-12-02 13:00:44 +00:00
|
|
|
|
import com.lagradost.cloudstream3.utils.ExtractorLink
|
|
|
|
|
import com.lagradost.cloudstream3.utils.Qualities
|
|
|
|
|
import com.lagradost.cloudstream3.utils.getQualityFromName
|
|
|
|
|
import com.lagradost.nicehttp.requestCreator
|
2022-12-07 15:06:48 +00:00
|
|
|
|
import okhttp3.FormBody
|
2022-12-02 13:00:44 +00:00
|
|
|
|
import okhttp3.HttpUrl.Companion.toHttpUrl
|
|
|
|
|
import java.net.URI
|
|
|
|
|
|
|
|
|
|
data class FilmxyCookies(
|
|
|
|
|
val phpsessid: String? = null,
|
|
|
|
|
val wLog: String? = null,
|
|
|
|
|
val wSec: String? = null,
|
|
|
|
|
)
|
|
|
|
|
|
2022-12-05 21:03:26 +00:00
|
|
|
|
fun String.filterIframe(seasonNum: Int?, lastSeason: Int?, year: Int?): Boolean {
|
2022-12-02 13:00:44 +00:00
|
|
|
|
return if (seasonNum != null) {
|
2022-12-05 21:03:26 +00:00
|
|
|
|
if (lastSeason == 1) {
|
2022-12-02 13:00:44 +00:00
|
|
|
|
this.contains(Regex("(?i)(S0?$seasonNum)|(Season\\s0?$seasonNum)|([0-9]{3,4}p)")) && !this.contains(
|
|
|
|
|
"Download",
|
|
|
|
|
true
|
|
|
|
|
)
|
|
|
|
|
} else {
|
|
|
|
|
this.contains(Regex("(?i)(S0?$seasonNum)|(Season\\s0?$seasonNum)")) && !this.contains(
|
|
|
|
|
"Download",
|
|
|
|
|
true
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
this.contains("$year", true) && !this.contains("Download", true)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fun String.filterMedia(title: String?, yearNum: Int?, seasonNum: Int?): Boolean {
|
|
|
|
|
return if (seasonNum != null) {
|
|
|
|
|
when {
|
|
|
|
|
seasonNum > 1 -> this.contains(Regex("(?i)(Season\\s0?1-0?$seasonNum)|(S0?1-S?0?$seasonNum)")) && this.contains(
|
|
|
|
|
"$title",
|
|
|
|
|
true
|
|
|
|
|
) && this.contains("$yearNum")
|
|
|
|
|
else -> this.contains(Regex("(?i)(Season\\s0?1)|(S0?1)")) && this.contains(
|
|
|
|
|
"$title",
|
|
|
|
|
true
|
|
|
|
|
) && this.contains("$yearNum")
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
this.contains("$title", true) && this.contains("$yearNum")
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-12-07 19:17:24 +00:00
|
|
|
|
suspend fun extractMirrorUHD(url: String, ref: String): String? {
|
|
|
|
|
val baseDoc = app.get(fixUrl(url, ref)).document
|
|
|
|
|
val downLink = baseDoc.select("div.mb-4 a").randomOrNull()
|
|
|
|
|
?.attr("href") ?: run {
|
|
|
|
|
val server = baseDoc.select("div.text-center a:contains(Server 2)").attr("href")
|
|
|
|
|
app.get(fixUrl(server, ref)).document.selectFirst("div.mb-4 a")
|
|
|
|
|
?.attr("href")
|
|
|
|
|
}
|
|
|
|
|
val downPage = app.get(downLink ?: return null).document
|
|
|
|
|
return downPage.selectFirst("form[method=post] a.btn.btn-success")
|
|
|
|
|
?.attr("onclick")?.substringAfter("Openblank('")?.substringBefore("')") ?: run {
|
|
|
|
|
val mirror = downPage.selectFirst("form[method=post] a.btn.btn-primary")
|
|
|
|
|
?.attr("onclick")?.substringAfter("Openblank('")?.substringBefore("')")
|
|
|
|
|
app.get(
|
|
|
|
|
mirror ?: return null
|
|
|
|
|
).document.selectFirst("script:containsData(input.value =)")
|
|
|
|
|
?.data()?.substringAfter("input.value = '")?.substringBefore("';")
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
suspend fun extractBackupUHD(url: String): String? {
|
|
|
|
|
val resumeDoc = app.get(url)
|
|
|
|
|
|
|
|
|
|
val script = resumeDoc.document.selectFirst("script:containsData(FormData.)")?.data()
|
|
|
|
|
|
|
|
|
|
val ssid = resumeDoc.cookies["PHPSESSID"]
|
|
|
|
|
val baseIframe = getBaseUrl(url)
|
|
|
|
|
val fetchLink = script?.substringAfter("fetch('")?.substringBefore("',")?.let { fixUrl(it, baseIframe) }
|
|
|
|
|
val token = script?.substringAfter("'token', '")?.substringBefore("');")
|
|
|
|
|
|
|
|
|
|
val body = FormBody.Builder()
|
|
|
|
|
.addEncoded("token", "$token")
|
|
|
|
|
.build()
|
|
|
|
|
val cookies = mapOf("PHPSESSID" to "$ssid")
|
|
|
|
|
|
|
|
|
|
val result = app.post(
|
|
|
|
|
fetchLink ?: return null,
|
|
|
|
|
requestBody = body,
|
|
|
|
|
headers = mapOf(
|
|
|
|
|
"Accept" to "*/*",
|
|
|
|
|
"Origin" to baseIframe,
|
|
|
|
|
"Sec-Fetch-Site" to "same-origin"
|
|
|
|
|
),
|
|
|
|
|
cookies = cookies,
|
|
|
|
|
referer = url
|
|
|
|
|
).text
|
|
|
|
|
return tryParseJson<UHDBackupUrl>(result)?.url
|
|
|
|
|
}
|
|
|
|
|
|
2022-12-02 16:50:56 +00:00
|
|
|
|
suspend fun extractGdbot(url: String): String? {
|
|
|
|
|
val headers = mapOf(
|
|
|
|
|
"Accept" to "text/html,application/xhtml+xml,application/xml;q=0.9,image/avif,image/webp,*/*;q=0.8",
|
|
|
|
|
)
|
|
|
|
|
val res = app.get(
|
|
|
|
|
"$gdbot/", headers = headers
|
|
|
|
|
)
|
|
|
|
|
val token = res.document.selectFirst("input[name=_token]")?.attr("value")
|
|
|
|
|
val cookiesSet = res.headers.filter { it.first == "set-cookie" }
|
2022-12-07 15:06:48 +00:00
|
|
|
|
val xsrf =
|
|
|
|
|
cookiesSet.find { it.second.contains("XSRF-TOKEN") }?.second?.substringAfter("XSRF-TOKEN=")
|
|
|
|
|
?.substringBefore(";")
|
|
|
|
|
val session =
|
|
|
|
|
cookiesSet.find { it.second.contains("gdtot_proxy_session") }?.second?.substringAfter("gdtot_proxy_session=")
|
|
|
|
|
?.substringBefore(";")
|
2022-12-02 16:50:56 +00:00
|
|
|
|
|
|
|
|
|
val cookies = mapOf(
|
|
|
|
|
"gdtot_proxy_session" to "$session",
|
|
|
|
|
"XSRF-TOKEN" to "$xsrf"
|
|
|
|
|
)
|
|
|
|
|
val requestFile = app.post(
|
|
|
|
|
"$gdbot/file", data = mapOf(
|
|
|
|
|
"link" to url,
|
|
|
|
|
"_token" to "$token"
|
|
|
|
|
), headers = headers, referer = "$gdbot/", cookies = cookies
|
|
|
|
|
).document
|
|
|
|
|
|
|
|
|
|
return requestFile.selectFirst("div.mt-8 a.float-right")?.attr("href")
|
|
|
|
|
}
|
|
|
|
|
|
2022-12-07 15:06:48 +00:00
|
|
|
|
suspend fun extractDrivebot(url: String): String? {
|
|
|
|
|
val iframeGdbot =
|
|
|
|
|
app.get(url).document.selectFirst("li.flex.flex-col.py-6 a:contains(Drivebot)")
|
|
|
|
|
?.attr("href")
|
|
|
|
|
val driveDoc = app.get(iframeGdbot ?: return null)
|
|
|
|
|
|
|
|
|
|
val ssid = driveDoc.cookies["PHPSESSID"]
|
|
|
|
|
val script = driveDoc.document.selectFirst("script:containsData(var formData)")?.data()
|
|
|
|
|
|
|
|
|
|
val baseUrl = getBaseUrl(iframeGdbot)
|
|
|
|
|
val token = script?.substringAfter("'token', '")?.substringBefore("');")
|
|
|
|
|
val link =
|
|
|
|
|
script?.substringAfter("fetch('")?.substringBefore("',").let { "$baseUrl$it" }
|
|
|
|
|
|
|
|
|
|
val body = FormBody.Builder()
|
|
|
|
|
.addEncoded("token", "$token")
|
|
|
|
|
.build()
|
|
|
|
|
val cookies = mapOf("PHPSESSID" to "$ssid")
|
|
|
|
|
|
|
|
|
|
val result = app.post(
|
|
|
|
|
link,
|
|
|
|
|
requestBody = body,
|
|
|
|
|
headers = mapOf(
|
|
|
|
|
"Accept" to "*/*",
|
|
|
|
|
"Origin" to baseUrl,
|
|
|
|
|
"Sec-Fetch-Site" to "same-origin"
|
|
|
|
|
),
|
|
|
|
|
cookies = cookies,
|
|
|
|
|
referer = iframeGdbot
|
|
|
|
|
).text
|
2022-12-07 19:17:24 +00:00
|
|
|
|
return tryParseJson<DriveBotLink>(result)?.url
|
2022-12-07 15:06:48 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
suspend fun bypassFdAds(url: String): String? {
|
|
|
|
|
val res = app.get(url).document
|
|
|
|
|
val freeRedirect = res.selectFirst("a#link")?.attr("href")
|
2022-12-07 19:17:24 +00:00
|
|
|
|
val res2 = app.get(freeRedirect ?: return null, verify = false).document
|
2022-12-07 15:06:48 +00:00
|
|
|
|
val formLink = res2.select("form#landing").attr("action")
|
|
|
|
|
val value = res2.select("form#landing input").attr("value")
|
|
|
|
|
val res3 = app.post(formLink, data = mapOf("go" to value), verify = false).document
|
|
|
|
|
val formLink2 = res3.select("form#landing").attr("action")
|
|
|
|
|
val humanVer = res3.select("form#landing input[name=humanverification]").attr("value")
|
|
|
|
|
val newwp = res3.select("form#landing input[name=newwpsafelink]").attr("value")
|
|
|
|
|
val res4 = app.post(
|
|
|
|
|
formLink2,
|
|
|
|
|
data = mapOf("humanverification" to humanVer, "newwpsafelink" to newwp),
|
|
|
|
|
verify = false
|
|
|
|
|
).document
|
|
|
|
|
val formLink3 = res4.select("form#wpsafelink-landing").attr("action")
|
|
|
|
|
val newwpsafelink =
|
|
|
|
|
res4.select("form#wpsafelink-landing input[name=newwpsafelink]").attr("value")
|
|
|
|
|
val res5 = app.post(
|
|
|
|
|
formLink3,
|
|
|
|
|
data = mapOf("newwpsafelink" to newwpsafelink),
|
|
|
|
|
verify = false
|
|
|
|
|
).document
|
|
|
|
|
val finalLink = res5.selectFirst("div#wpsafe-link a")?.attr("onclick")?.substringAfter("open('")
|
|
|
|
|
?.substringBefore("',")
|
|
|
|
|
return app.get(finalLink ?: return null, verify = false).url
|
|
|
|
|
}
|
|
|
|
|
|
2022-12-02 13:00:44 +00:00
|
|
|
|
suspend fun getFilmxyCookies(imdbId: String? = null, season: Int? = null): FilmxyCookies? {
|
|
|
|
|
|
|
|
|
|
val url = if (season == null) {
|
2022-12-07 19:17:24 +00:00
|
|
|
|
"${filmxyAPI}/movie/$imdbId"
|
2022-12-02 13:00:44 +00:00
|
|
|
|
} else {
|
2022-12-07 19:17:24 +00:00
|
|
|
|
"${filmxyAPI}/tv/$imdbId"
|
2022-12-02 13:00:44 +00:00
|
|
|
|
}
|
2022-12-07 19:17:24 +00:00
|
|
|
|
val cookieUrl = "${filmxyAPI}/wp-admin/admin-ajax.php"
|
2022-12-02 13:00:44 +00:00
|
|
|
|
|
|
|
|
|
val res = session.get(
|
|
|
|
|
url,
|
|
|
|
|
headers = mapOf(
|
|
|
|
|
"Accept" to "text/html,application/xhtml+xml,application/xml;q=0.9,image/avif,image/webp,*/*;q=0.8"
|
|
|
|
|
),
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
if (!res.isSuccessful) return FilmxyCookies()
|
|
|
|
|
|
|
|
|
|
val userNonce =
|
|
|
|
|
res.document.select("script").find { it.data().contains("var userNonce") }?.data()?.let {
|
|
|
|
|
Regex("var\\suserNonce.*?\"(\\S+?)\";").find(it)?.groupValues?.get(1)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var phpsessid = session.baseClient.cookieJar.loadForRequest(url.toHttpUrl())
|
|
|
|
|
.first { it.name == "PHPSESSID" }.value
|
|
|
|
|
|
|
|
|
|
session.post(
|
|
|
|
|
cookieUrl,
|
|
|
|
|
data = mapOf(
|
|
|
|
|
"action" to "guest_login",
|
|
|
|
|
"nonce" to "$userNonce",
|
|
|
|
|
),
|
|
|
|
|
headers = mapOf(
|
|
|
|
|
"Cookie" to "PHPSESSID=$phpsessid; G_ENABLED_IDPS=google",
|
|
|
|
|
"X-Requested-With" to "XMLHttpRequest",
|
|
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
val cookieJar = session.baseClient.cookieJar.loadForRequest(cookieUrl.toHttpUrl())
|
|
|
|
|
phpsessid = cookieJar.first { it.name == "PHPSESSID" }.value
|
|
|
|
|
val wLog =
|
|
|
|
|
cookieJar.first { it.name == "wordpress_logged_in_8bf9d5433ac88cc9a3a396d6b154cd01" }.value
|
|
|
|
|
val wSec = cookieJar.first { it.name == "wordpress_sec_8bf9d5433ac88cc9a3a396d6b154cd01" }.value
|
|
|
|
|
|
|
|
|
|
return FilmxyCookies(phpsessid, wLog, wSec)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fun String?.fixTitle(): String? {
|
|
|
|
|
return this?.replace(Regex("[!%:]|( &)"), "")?.replace(" ", "-")?.lowercase()
|
|
|
|
|
?.replace("-–-", "-")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fun getLanguage(str: String): String {
|
|
|
|
|
return if (str.contains("(in_ID)")) "Indonesian" else str
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fun getKisskhTitle(str: String?): String? {
|
|
|
|
|
return str?.replace(Regex("[^a-zA-Z0-9]"), "-")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fun getQuality(str: String): Int {
|
|
|
|
|
return when (str) {
|
|
|
|
|
"360p" -> Qualities.P240.value
|
|
|
|
|
"480p" -> Qualities.P360.value
|
|
|
|
|
"720p" -> Qualities.P480.value
|
|
|
|
|
"1080p" -> Qualities.P720.value
|
|
|
|
|
"1080p Ultra" -> Qualities.P1080.value
|
|
|
|
|
else -> getQualityFromName(str)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-12-02 16:50:56 +00:00
|
|
|
|
fun getGMoviesQuality(str: String): Int {
|
|
|
|
|
return when {
|
|
|
|
|
str.contains("480P", true) -> Qualities.P480.value
|
|
|
|
|
str.contains("720P", true) -> Qualities.P720.value
|
|
|
|
|
str.contains("1080", true) -> Qualities.P1080.value
|
|
|
|
|
str.contains("4K", true) -> Qualities.P2160.value
|
|
|
|
|
else -> Qualities.Unknown.value
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-12-02 13:00:44 +00:00
|
|
|
|
fun getBaseUrl(url: String): String {
|
|
|
|
|
return URI(url).let {
|
|
|
|
|
"${it.scheme}://${it.host}"
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fun decryptStreamUrl(data: String): String {
|
|
|
|
|
|
|
|
|
|
fun getTrash(arr: List<String>, item: Int): List<String> {
|
|
|
|
|
val trash = ArrayList<List<String>>()
|
|
|
|
|
for (i in 1..item) {
|
|
|
|
|
trash.add(arr)
|
|
|
|
|
}
|
|
|
|
|
return trash.reduce { acc, list ->
|
|
|
|
|
val temp = ArrayList<String>()
|
|
|
|
|
acc.forEach { ac ->
|
|
|
|
|
list.forEach { li ->
|
|
|
|
|
temp.add(ac.plus(li))
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return@reduce temp
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
val trashList = listOf("@", "#", "!", "^", "$")
|
|
|
|
|
val trashSet = getTrash(trashList, 2) + getTrash(trashList, 3)
|
|
|
|
|
var trashString = data.replace("#2", "").split("//_//").joinToString("")
|
|
|
|
|
|
|
|
|
|
trashSet.forEach {
|
|
|
|
|
val temp = base64Encode(it.toByteArray())
|
|
|
|
|
trashString = trashString.replace(temp, "")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return base64Decode(trashString)
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fun fixUrl(url: String, domain: String): String {
|
|
|
|
|
if (url.startsWith("http")) {
|
|
|
|
|
return url
|
|
|
|
|
}
|
|
|
|
|
if (url.isEmpty()) {
|
|
|
|
|
return ""
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
val startsWithNoHttp = url.startsWith("//")
|
|
|
|
|
if (startsWithNoHttp) {
|
|
|
|
|
return "https:$url"
|
|
|
|
|
} else {
|
|
|
|
|
if (url.startsWith('/')) {
|
|
|
|
|
return domain + url
|
|
|
|
|
}
|
|
|
|
|
return "$domain/$url"
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
suspend fun loadLinksWithWebView(
|
|
|
|
|
url: String,
|
|
|
|
|
callback: (ExtractorLink) -> Unit
|
|
|
|
|
) {
|
|
|
|
|
val foundVideo = WebViewResolver(
|
|
|
|
|
Regex("""\.m3u8|i7njdjvszykaieynzsogaysdgb0hm8u1mzubmush4maopa4wde\.com""")
|
|
|
|
|
).resolveUsingWebView(
|
|
|
|
|
requestCreator(
|
|
|
|
|
"GET", url, referer = "https://olgply.com/"
|
|
|
|
|
)
|
|
|
|
|
).first ?: return
|
|
|
|
|
|
|
|
|
|
callback.invoke(
|
|
|
|
|
ExtractorLink(
|
|
|
|
|
"Olgply",
|
|
|
|
|
"Olgply",
|
|
|
|
|
foundVideo.url.toString(),
|
|
|
|
|
"",
|
|
|
|
|
Qualities.P1080.value,
|
|
|
|
|
true
|
|
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
}
|