SussyToons: Fixes (#7211)

* Fix open http connection, dto serialization and loading chapter and pages

* Change message

* Save the last url of the chapter script

* Remove unused class

* Remove duplicate code

* Fix typo
This commit is contained in:
Chopper 2025-01-17 12:28:50 -03:00 committed by Draff
parent 1b3368110f
commit 554872b754
No known key found for this signature in database
GPG Key ID: E8A89F3211677653
3 changed files with 159 additions and 76 deletions

View File

@ -1,7 +1,7 @@
ext { ext {
extName = 'Sussy Toons' extName = 'Sussy Toons'
extClass = '.SussyToons' extClass = '.SussyToons'
extVersionCode = 47 extVersionCode = 48
isNsfw = true isNsfw = true
} }

View File

@ -7,6 +7,7 @@ import android.os.Handler
import android.os.Looper import android.os.Looper
import android.webkit.WebResourceRequest import android.webkit.WebResourceRequest
import android.webkit.WebResourceResponse import android.webkit.WebResourceResponse
import android.webkit.WebSettings
import android.webkit.WebView import android.webkit.WebView
import android.webkit.WebViewClient import android.webkit.WebViewClient
import android.widget.Toast import android.widget.Toast
@ -37,7 +38,6 @@ import uy.kohesive.injekt.Injekt
import uy.kohesive.injekt.api.get import uy.kohesive.injekt.api.get
import uy.kohesive.injekt.injectLazy import uy.kohesive.injekt.injectLazy
import java.io.IOException import java.io.IOException
import java.net.SocketTimeoutException
import java.text.SimpleDateFormat import java.text.SimpleDateFormat
import java.util.Locale import java.util.Locale
import java.util.concurrent.CountDownLatch import java.util.concurrent.CountDownLatch
@ -63,23 +63,21 @@ class SussyToons : HttpSource(), ConfigurableSource {
private val preferences: SharedPreferences = private val preferences: SharedPreferences =
Injekt.get<Application>().getSharedPreferences("source_$id", 0x0000) Injekt.get<Application>().getSharedPreferences("source_$id", 0x0000)
private var _apiUrlCache: String? = null
private var apiUrl: String private var apiUrl: String
get() = _apiUrlCache ?: preferences.prefApiUrl.also { _apiUrlCache = it } get() = preferences.getString(API_BASE_URL_PREF, defaultApiUrl)!!
set(value) { _apiUrlCache = value } set(value) = preferences.edit().putString(API_BASE_URL_PREF, value).apply()
private var chapterScriptUrl: String
get() = preferences.getString(CHAPTER_SCRIPT_URL_PREF, "")!!
set(value) = preferences.edit().putString(CHAPTER_SCRIPT_URL_PREF, value).apply()
private var pageScriptUrl: String
get() = preferences.getString(PAGE_SCRIPT_URL_PREF, "")!!
set(value) = preferences.edit().putString(PAGE_SCRIPT_URL_PREF, value).apply()
override val baseUrl: String get() = when { override val baseUrl: String get() = when {
isCi -> defaultBaseUrl isCi -> defaultBaseUrl
else -> preferences.prefBaseUrl else -> preferences.getString(BASE_URL_PREF, defaultBaseUrl)!!
}
private val SharedPreferences.prefBaseUrl: String get() = getString(BASE_URL_PREF, defaultBaseUrl)!!
private val SharedPreferences.prefApiUrl: String get() = getString(API_BASE_URL_PREF, defaultApiUrl)!!
private fun SharedPreferences.prefApiUrlUpSet(url: String): String {
edit().putString(API_BASE_URL_PREF, url)
.apply()
return url
} }
private val defaultBaseUrl: String = "https://www.sussytoons.site" private val defaultBaseUrl: String = "https://www.sussytoons.site"
@ -213,10 +211,10 @@ class SussyToons : HttpSource(), ConfigurableSource {
} }
} }
private var pageUrl: String? = null private var pageUrlSegment: String? = null
override fun pageListParse(response: Response): List<Page> { override fun pageListParse(response: Response): List<Page> {
pageUrl = pageUrl ?: findPageUrl(response) pageUrlSegment = pageUrlSegment ?: findPageUrlSegment(response)
val chapterPageId = response.request.url.pathSegments.last() val chapterPageId = response.request.url.pathSegments.last()
val chapterUrl = response.request.url.fragment val chapterUrl = response.request.url.fragment
@ -224,7 +222,7 @@ class SussyToons : HttpSource(), ConfigurableSource {
?: throw Exception("Não foi possivel carregar as páginas") ?: throw Exception("Não foi possivel carregar as páginas")
val url = apiUrl.toHttpUrl().newBuilder() val url = apiUrl.toHttpUrl().newBuilder()
.addEncodedPathSegments(pageUrl!!) .addEncodedPathSegments(pageUrlSegment!!)
.addPathSegment(chapterPageId) .addPathSegment(chapterPageId)
.fragment( .fragment(
"$chapterPagePrefix${"$baseUrl$chapterUrl"}", "$chapterPagePrefix${"$baseUrl$chapterUrl"}",
@ -254,16 +252,23 @@ class SussyToons : HttpSource(), ConfigurableSource {
/** /**
* Get the dynamic path segment of the chapter page * Get the dynamic path segment of the chapter page
*/ */
private fun findPageUrl(response: Response): String { private fun findPageUrlSegment(response: Response): String {
val document = response.asJsoup() val scriptUrls = when {
val scriptUrl = document.select("script[src]") pageScriptUrl.isNotBlank() -> listOf(pageScriptUrl to headers)
.map { it.absUrl("src") } else -> emptyList()
.firstOrNull { it.contains("app/capitulo", ignoreCase = true) } }
?: throw IOException("Não foi possivel encontrar a URL da página")
return client.newCall(GET(scriptUrl, headers)).execute().use { val script = loadJsScript(
pageUrlRegex.find(it.body.string())?.groups?.get(1)?.value?.toPathSegment() urls = scriptUrls,
} ?: throw IOException("Não foi possivel extrair a URL da página") doRequest = { client.newCall(it).execute() },
pattern = pageUrlRegex,
fallback = { fetchAllNextJsScriptUrls(response.request) },
)
pageScriptUrl = script.url
return pageUrlRegex.find(script.body)?.groups?.get(2)?.value?.toPathSegment()
?: throw IOException("Não foi encontrar o caminho das páginas")
} }
override fun imageUrlParse(response: Response): String = "" override fun imageUrlParse(response: Response): String = ""
@ -285,15 +290,11 @@ class SussyToons : HttpSource(), ConfigurableSource {
val request = chain.request() val request = chain.request()
val response: Response = try { val response: Response = try {
chain.proceed(request) chain.proceed(request)
} catch (ex: SocketTimeoutException) { } catch (ex: Exception) {
chain.createTimeoutResponse(request) chain.createBadGatewayResponse(request)
} }
if (request.url.toString().contains(apiUrl).not()) { if (response.isSuccessful || request.url.toString().contains(apiUrl).not()) {
return response
}
if (response.isSuccessful) {
return response return response
} }
@ -308,9 +309,13 @@ class SussyToons : HttpSource(), ConfigurableSource {
.url(url) .url(url)
.build() .build()
return chain.proceed(newRequest).takeIf(Response::isSuccessful).also { val localResponse = chain.proceed(newRequest)
apiUrl = preferences.prefApiUrlUpSet(urlCandidate) if (localResponse.isSuccessful.not()) {
} ?: return@forEach localResponse.close()
return@forEach
}
apiUrl = urlCandidate
return localResponse
} }
throw IOException( throw IOException(
@ -320,7 +325,8 @@ class SussyToons : HttpSource(), ConfigurableSource {
}, },
) )
} }
private fun Interceptor.Chain.createTimeoutResponse(request: Request): Response {
private fun Interceptor.Chain.createBadGatewayResponse(request: Request): Response {
return Response.Builder() return Response.Builder()
.request(request) .request(request)
.protocol(Protocol.HTTP_1_1) .protocol(Protocol.HTTP_1_1)
@ -366,22 +372,45 @@ class SussyToons : HttpSource(), ConfigurableSource {
}?.substringAfter(mangaPagePrefix) }?.substringAfter(mangaPagePrefix)
?: return chain.proceed(request) ?: return chain.proceed(request)
val document = chain.proceed(GET(mangaUrl, headers)).asJsoup() val scriptUrls = when {
chapterScriptUrl.isNotBlank() -> listOf(chapterScriptUrl to headers)
val scriptUrl = document.select("script[src]") else -> emptyList()
.map { it.absUrl("src") }
.firstOrNull { it.contains("app/obra", ignoreCase = true) }
?: throw IOException("Não foi possivel encontrar a URL do capitulo")
chapterUrl = chain.proceed(GET(scriptUrl, headers)).use { response ->
response.body.string().let {
chapterUrlRegex.find(it)?.groups?.get(1)?.value?.toPathSegment()
} ?: throw IOException("Não foi possivel extrair a URL do capitulo")
} }
val script = loadJsScript(
urls = scriptUrls,
doRequest = chain::proceed,
pattern = chapterUrlRegex,
fallback = { fetchAllNextJsScriptUrls(GET(mangaUrl, headers)) },
)
chapterScriptUrl = script.url
chapterUrl = chapterUrlRegex.find(script.body)?.groups?.get(1)?.value?.toPathSegment()
?: throw IOException("Não foi possivel extrair a URL do capitulo")
return chain.proceed(request) return chain.proceed(request)
} }
private fun loadJsScript(
urls: List<Pair<String, Headers>>,
doRequest: (request: Request) -> Response,
pattern: Regex,
fallback: (() -> List<Pair<String, Headers>>)? = null,
): Script {
val script = urls.map { pair ->
val request = GET(pair.first, pair.second)
Script(
url = request.url.toString(),
body = doRequest(request).use { response -> response.body.string() },
)
}.firstOrNull { pattern.containsMatchIn(it.body) }
return script ?: fallback?.let { urlList ->
loadJsScript(urlList(), doRequest = doRequest, pattern = pattern)
} ?: throw IOException("Não foi possivel encontrar a URL do capitulo")
}
private fun imageLocation(chain: Interceptor.Chain): Response { private fun imageLocation(chain: Interceptor.Chain): Response {
val request = chain.request() val request = chain.request()
val response = chain.proceed(request) val response = chain.proceed(request)
@ -438,16 +467,76 @@ class SussyToons : HttpSource(), ConfigurableSource {
@SuppressLint("SetJavaScriptEnabled") @SuppressLint("SetJavaScriptEnabled")
private fun fetchChapterPagesHeaders(baseRequest: Request, originRequest: Request): Request { private fun fetchChapterPagesHeaders(baseRequest: Request, originRequest: Request): Request {
fun WebResourceRequest.isOriginRequest() =
originRequest.url.toString().equals(this.url.toString(), ignoreCase = true)
chapterPageHeaders = handlingWithWebResourceRequest(
baseRequest,
initial = headersBuilder(),
stopCondition = { _, _, resource ->
resource.isOriginRequest() && resource.method.equals("GET", true)
},
fold = { headers, _, resource ->
headers.apply {
if (resource.isOriginRequest().not() || resource.method.equals("GET", true).not()) {
return@apply
}
fill(resource.requestHeaders)
}
},
).build()
return originRequest.newBuilder()
.headers(chapterPageHeaders!!)
.build()
}
@SuppressLint("SetJavaScriptEnabled")
private fun fetchAllNextJsScriptUrls(baseRequest: Request): List<Pair<String, Headers>> {
fun WebResourceRequest.isNextJSUrl() = this.url.toString().contains("_next", ignoreCase = true) &&
this.url.toString().contains(".js", ignoreCase = true)
return handlingWithWebResourceRequest(
baseRequest,
initial = mutableListOf(),
stopCondition = { urls, _, _ ->
val minUrlsAvailable = 24
urls.size > minUrlsAvailable
},
fold = { urls, base, resource ->
urls.apply {
if (resource.isNextJSUrl().not()) {
return@apply
}
val headers = base.headers.newBuilder().apply {
fill(resource.requestHeaders)
}
add(resource.url.toString() to headers.build())
}
},
)
}
@SuppressLint("SetJavaScriptEnabled")
private fun <T> handlingWithWebResourceRequest(
baseRequest: Request,
initial: T,
stopCondition: (T, Request, WebResourceRequest) -> Boolean,
fold: (T, Request, WebResourceRequest) -> T,
): T {
val latch = CountDownLatch(1) val latch = CountDownLatch(1)
val headers = originRequest.headers.newBuilder()
var webView: WebView? = null var webView: WebView? = null
val looper = Handler(Looper.getMainLooper()) val looper = Handler(Looper.getMainLooper())
var state = initial
looper.post { looper.post {
webView = WebView(Injekt.get<Application>()) webView = WebView(Injekt.get<Application>())
webView?.let { webView?.let {
with(it.settings) { with(it.settings) {
javaScriptEnabled = true javaScriptEnabled = true
blockNetworkImage = true domStorageEnabled = true
useWideViewPort = true
loadWithOverviewMode = true
cacheMode = WebSettings.LOAD_DEFAULT
} }
} }
webView?.webViewClient = object : WebViewClient() { webView?.webViewClient = object : WebViewClient() {
@ -455,23 +544,14 @@ class SussyToons : HttpSource(), ConfigurableSource {
view: WebView?, view: WebView?,
request: WebResourceRequest, request: WebResourceRequest,
): WebResourceResponse? { ): WebResourceResponse? {
val ignore = listOf(".css", "google", "fonts", "ads") state = fold(state, baseRequest, request)
val url = request.url.toString() if (stopCondition(state, baseRequest, request)) {
if (ignore.any { url.contains(it, ignoreCase = true) }) {
return emptyResource()
}
if (request.isOriginRequest() && request.method.equals("GET", true)) {
headers.fill(request.requestHeaders)
latch.countDown() latch.countDown()
} }
return super.shouldInterceptRequest(view, request) return super.shouldInterceptRequest(view, request)
} }
private fun WebResourceRequest.isOriginRequest() =
originRequest.url.toString().equals(this.url.toString(), ignoreCase = true)
private fun emptyResource() = WebResourceResponse(null, null, null)
} }
webView?.loadUrl(baseRequest.url.toString(), headers.build().toMap()) webView?.loadUrl(baseRequest.url.toString(), baseRequest.headers.toMap())
} }
latch.await(client.readTimeoutMillis.toLong(), TimeUnit.MILLISECONDS) latch.await(client.readTimeoutMillis.toLong(), TimeUnit.MILLISECONDS)
@ -482,12 +562,7 @@ class SussyToons : HttpSource(), ConfigurableSource {
destroy() destroy()
} }
} }
return state
chapterPageHeaders = headers.build()
return originRequest.newBuilder()
.headers(chapterPageHeaders!!)
.build()
} }
private fun Headers.Builder.fill(from: Map<String, String>): Headers.Builder { private fun Headers.Builder.fill(from: Map<String, String>): Headers.Builder {
@ -540,6 +615,11 @@ class SussyToons : HttpSource(), ConfigurableSource {
// ============================= Utilities ==================================== // ============================= Utilities ====================================
class Script(
val url: String,
val body: String,
)
private fun MangaDto.toSManga(): SManga { private fun MangaDto.toSManga(): SManga {
val sManga = SManga.create().apply { val sManga = SManga.create().apply {
title = name title = name
@ -552,7 +632,7 @@ class SussyToons : HttpSource(), ConfigurableSource {
setUrlWithoutDomain(mangaUrl.toString()) setUrlWithoutDomain(mangaUrl.toString())
} }
Jsoup.parseBodyFragment(description).let { sManga.description = it.text() } description?.let { Jsoup.parseBodyFragment(it).let { sManga.description = it.text() } }
sManga.status = status.toStatus() sManga.status = status.toStatus()
return sManga return sManga
@ -592,8 +672,11 @@ class SussyToons : HttpSource(), ConfigurableSource {
private const val API_BASE_URL_PREF_TITLE = "Editar URL da API da fonte" private const val API_BASE_URL_PREF_TITLE = "Editar URL da API da fonte"
private const val API_DEFAULT_BASE_URL_PREF = "defaultApiUrl" private const val API_DEFAULT_BASE_URL_PREF = "defaultApiUrl"
val chapterUrlRegex = """push\("([^"]*capitulo[^"]*)/?"\.concat""".toRegex() private const val CHAPTER_SCRIPT_URL_PREF = "chapterScriptUrl"
val pageUrlRegex = """\.get\("([^"]*capitulo[^(/?")]*)/?"\.concat""".toRegex() private const val PAGE_SCRIPT_URL_PREF = "pageScriptUrl"
val chapterUrlRegex = """push\("([^"]*capitulo[^"]*)\/?"\.concat""".toRegex()
val pageUrlRegex = """\.(get|post)\("([^"]*capitulo[^"]*)\/?"\.concat""".toRegex()
val apiUrlRegex = """(?<=production",)(.*?)(?=;function)""".toRegex() val apiUrlRegex = """(?<=production",)(.*?)(?=;function)""".toRegex()
val urlRegex = """https?://[\w\-]+(\.[\w\-]+)+[/#?]?.*$""".toRegex() val urlRegex = """https?://[\w\-]+(\.[\w\-]+)+[/#?]?.*$""".toRegex()

View File

@ -24,9 +24,9 @@ class MangaDto(
@SerialName("obr_id") @SerialName("obr_id")
val id: Int, val id: Int,
@SerialName("obr_descricao") @SerialName("obr_descricao")
val description: String, val description: String?,
@SerialName("obr_imagem") @SerialName("obr_imagem")
val thumbnail: String, val thumbnail: String?,
@SerialName("obr_nome") @SerialName("obr_nome")
val name: String, val name: String,
@SerialName("obr_slug") @SerialName("obr_slug")
@ -37,10 +37,10 @@ class MangaDto(
@Serializable @Serializable
class MangaStatus( class MangaStatus(
@SerialName("stt_nome") @SerialName("stt_nome")
val value: String, val value: String?,
) { ) {
fun toStatus(): Int { fun toStatus(): Int {
return when (value.lowercase()) { return when (value?.lowercase()) {
"em andamento" -> SManga.ONGOING "em andamento" -> SManga.ONGOING
"completo" -> SManga.COMPLETED "completo" -> SManga.COMPLETED
"hiato" -> SManga.ON_HIATUS "hiato" -> SManga.ON_HIATUS