diff --git a/src/all/hentaihand/build.gradle b/src/all/hentaihand/build.gradle index 53d68503d..69f1f3dd2 100644 --- a/src/all/hentaihand/build.gradle +++ b/src/all/hentaihand/build.gradle @@ -1,11 +1,12 @@ apply plugin: 'com.android.application' apply plugin: 'kotlin-android' +apply plugin: 'kotlinx-serialization' ext { extName = 'HentaiHand' pkgNameSuffix = 'all.hentaihand' extClass = '.HentaiHandFactory' - extVersionCode = 3 + extVersionCode = 4 libVersion = '1.2' containsNsfw = true } diff --git a/src/all/hentaihand/src/eu/kanade/tachiyomi/extension/all/hentaihand/HentaiHand.kt b/src/all/hentaihand/src/eu/kanade/tachiyomi/extension/all/hentaihand/HentaiHand.kt index 69ef4f322..9a935ff65 100644 --- a/src/all/hentaihand/src/eu/kanade/tachiyomi/extension/all/hentaihand/HentaiHand.kt +++ b/src/all/hentaihand/src/eu/kanade/tachiyomi/extension/all/hentaihand/HentaiHand.kt @@ -1,17 +1,9 @@ package eu.kanade.tachiyomi.extension.all.hentaihand -import android.annotation.SuppressLint import android.app.Application import android.content.SharedPreferences import android.text.InputType import android.widget.Toast -import com.github.salomonbrys.kotson.fromJson -import com.github.salomonbrys.kotson.get -import com.github.salomonbrys.kotson.nullObj -import com.github.salomonbrys.kotson.nullString -import com.google.gson.Gson -import com.google.gson.JsonArray -import com.google.gson.JsonObject import eu.kanade.tachiyomi.annotations.Nsfw import eu.kanade.tachiyomi.network.GET import eu.kanade.tachiyomi.network.POST @@ -24,20 +16,28 @@ import eu.kanade.tachiyomi.source.model.Page import eu.kanade.tachiyomi.source.model.SChapter import eu.kanade.tachiyomi.source.model.SManga import eu.kanade.tachiyomi.source.online.HttpSource +import kotlinx.serialization.json.Json +import kotlinx.serialization.json.JsonObject +import kotlinx.serialization.json.buildJsonObject +import kotlinx.serialization.json.int +import kotlinx.serialization.json.jsonArray +import kotlinx.serialization.json.jsonObject +import kotlinx.serialization.json.jsonPrimitive +import kotlinx.serialization.json.put import okhttp3.HttpUrl.Companion.toHttpUrlOrNull import okhttp3.Interceptor import okhttp3.MediaType.Companion.toMediaTypeOrNull import okhttp3.OkHttpClient import okhttp3.Request -import okhttp3.RequestBody +import okhttp3.RequestBody.Companion.toRequestBody import okhttp3.Response -import org.json.JSONException -import org.json.JSONObject import rx.Observable import rx.schedulers.Schedulers import uy.kohesive.injekt.Injekt import uy.kohesive.injekt.api.get +import uy.kohesive.injekt.injectLazy import java.text.SimpleDateFormat +import java.util.Locale @Nsfw abstract class HentaiHand( @@ -50,29 +50,37 @@ abstract class HentaiHand( override val name: String = "HentaiHand$extraName" override val supportsLatest = true - private val gson = Gson() - override val client: OkHttpClient = network.cloudflareClient.newBuilder() .addInterceptor { authIntercept(it) } .build() - private fun parseGenericResponse(response: Response): MangasPage { - val data = gson.fromJson(response.body!!.string()) - return MangasPage( - data.getAsJsonArray("data").map { - SManga.create().apply { - url = "/en/comic/${it["slug"].asString}" - title = it["title"].asString - thumbnail_url = it["thumb_url"].asString - } - }, - !data["next_page_url"].isJsonNull - ) + private val json: Json by injectLazy() + + private fun slugToUrl(json: JsonObject) = json["slug"]!!.jsonPrimitive.content.prependIndent("/en/comic/") + + private fun jsonArrayToString(arrayKey: String, obj: JsonObject): String? { + val array = obj[arrayKey]!!.jsonArray + if (array.isEmpty()) return null + return array.joinToString(", ") { + it.jsonObject["name"]!!.jsonPrimitive.content + } } // Popular - override fun popularMangaParse(response: Response): MangasPage = parseGenericResponse(response) + override fun popularMangaParse(response: Response): MangasPage { + val jsonResponse = json.parseToJsonElement(response.body!!.string()) + val mangaList = jsonResponse.jsonObject["data"]!!.jsonArray.map { + val obj = it.jsonObject + SManga.create().apply { + url = slugToUrl(obj) + title = obj["title"]!!.jsonPrimitive.content + thumbnail_url = obj["thumb_url"]!!.jsonPrimitive.content + } + } + val hasNextPage = jsonResponse.jsonObject["next_page_url"]!!.jsonPrimitive.content.isNotEmpty() + return MangasPage(mangaList, hasNextPage) + } override fun popularMangaRequest(page: Int): Request { val url = "$baseUrl/api/comics?page=$page&sort=popularity&order=desc&duration=all" @@ -81,7 +89,7 @@ abstract class HentaiHand( // Latest - override fun latestUpdatesParse(response: Response): MangasPage = parseGenericResponse(response) + override fun latestUpdatesParse(response: Response): MangasPage = popularMangaParse(response) override fun latestUpdatesRequest(page: Int): Request { val url = "$baseUrl/api/comics?page=$page&sort=uploaded_at&order=desc&duration=week" @@ -90,17 +98,20 @@ abstract class HentaiHand( // Search - override fun searchMangaParse(response: Response): MangasPage = parseGenericResponse(response) + override fun searchMangaParse(response: Response): MangasPage = popularMangaParse(response) private fun lookupFilterId(query: String, uri: String): Int? { // filter query needs to be resolved to an ID return client.newCall(GET("$baseUrl/api/$uri?q=$query")) .asObservableSuccess() .subscribeOn(Schedulers.io()) - .map { - val data = gson.fromJson(it.body!!.string()) - // only the first tag will be used - data.getAsJsonArray("data").firstOrNull()?.let { t -> t["id"].asInt } + .map { response -> + // Returns the first matched id, or null if there are no results + val idList = json.parseToJsonElement(response.body!!.string()).jsonObject["data"]!!.jsonArray.map { + it.jsonObject["id"]!!.jsonPrimitive.content + } + if (idList.isEmpty()) return@map null + else idList.first().toInt() }.toBlocking().first() } @@ -138,12 +149,6 @@ abstract class HentaiHand( // Details - private fun tagArrayToString(array: JsonArray, key: String = "name"): String? { - if (array.size() == 0) - return null - return array.joinToString { it[key].asString } - } - private fun mangaDetailsApiRequest(manga: SManga): Request { val slug = manga.url.removePrefix("/en/comic/") return GET("$baseUrl/api/comics/$slug") @@ -156,28 +161,26 @@ abstract class HentaiHand( } override fun mangaDetailsParse(response: Response): SManga { - val data = gson.fromJson(response.body!!.string()) + val obj = json.parseToJsonElement(response.body!!.string()).jsonObject return SManga.create().apply { - - artist = tagArrayToString(data.getAsJsonArray("artists")) - author = tagArrayToString(data.getAsJsonArray("authors")) ?: artist - - genre = listOf("tags", "relationships").map { - data.getAsJsonArray(it).map { t -> t["name"].asString } - }.flatten().distinct().joinToString() - + url = slugToUrl(obj) + title = obj["title"]!!.jsonPrimitive.content + thumbnail_url = obj["thumb_url"]!!.jsonPrimitive.content + artist = jsonArrayToString("artists", obj) + author = jsonArrayToString("authors", obj) ?: artist + genre = listOfNotNull(jsonArrayToString("tags", obj), jsonArrayToString("relationships", obj)).joinToString(", ") status = SManga.COMPLETED description = listOf( - Pair("Alternative Title", data["alternative_title"].nullString), - Pair("Groups", tagArrayToString(data.getAsJsonArray("groups"))), - Pair("Description", data["description"].nullString), - Pair("Pages", data["pages"].asInt.toString()), - Pair("Category", data["category"].nullObj?.get("name")?.asString), - Pair("Language", data["language"].nullObj?.get("name")?.asString), - Pair("Parodies", tagArrayToString(data.getAsJsonArray("parodies"))), - Pair("Characters", tagArrayToString(data.getAsJsonArray("characters"))) - ).filter { !it.second.isNullOrEmpty() }.joinToString("\n\n") { "${it.first}:\n${it.second}" } + Pair("Alternative Title", obj["alternative_title"]!!.jsonPrimitive.content), + Pair("Groups", jsonArrayToString("groups", obj)), + Pair("Description", obj["description"]!!.jsonPrimitive.content), + Pair("Pages", obj["pages"]!!.jsonPrimitive.content), + Pair("Category", obj["category"]!!.jsonObject["name"]!!.jsonPrimitive.content), + Pair("Language", obj["language"]!!.jsonObject["name"]!!.jsonPrimitive.content), + Pair("Parodies", jsonArrayToString("parodies", obj)), + Pair("Characters", jsonArrayToString("characters", obj)) + ).filter { !it.second.isNullOrEmpty() }.joinToString("\n\n") { "${it.first}: ${it.second}" } } } @@ -186,12 +189,12 @@ abstract class HentaiHand( override fun chapterListRequest(manga: SManga): Request = mangaDetailsApiRequest(manga) override fun chapterListParse(response: Response): List { - val data = gson.fromJson(response.body!!.string()) + val obj = json.parseToJsonElement(response.body!!.string()).jsonObject return listOf( SChapter.create().apply { - url = "/en/comic/${data["slug"].asString}/reader/1" + url = "/en/comic/${obj["slug"]!!.jsonPrimitive.content}/reader/1" name = "Chapter" - date_upload = DATE_FORMAT.parse(data["uploaded_at"].asString)?.time ?: 0 + date_upload = DATE_FORMAT.parse(obj["uploaded_at"]!!.jsonPrimitive.content)?.time ?: 0 chapter_number = 1f } ) @@ -204,12 +207,13 @@ abstract class HentaiHand( return GET("$baseUrl/api/comics/$slug/images") } - override fun pageListParse(response: Response): List { - val data = gson.fromJson(response.body!!.string()) - return data.getAsJsonArray("images").mapIndexed { i, it -> - Page(i, "/en/comic/${data["comic"]["slug"].asString}/reader/${it["page"].asInt}", it["source_url"].asString) + override fun pageListParse(response: Response): List = + json.parseToJsonElement(response.body!!.string()).jsonObject["images"]!!.jsonArray.map { + val imgObj = it.jsonObject + val index = imgObj["page"]!!.jsonPrimitive.int + val imgUrl = imgObj["source_url"]!!.jsonPrimitive.content + Page(index, "", imgUrl) } - } override fun imageUrlParse(response: Response): String = throw UnsupportedOperationException("Not used") @@ -231,12 +235,12 @@ abstract class HentaiHand( } private fun login(chain: Interceptor.Chain, username: String, password: String): String { - val jsonObject = JSONObject().apply { - this.put("username", username) - this.put("password", password) - this.put("remember_me", true) + val jsonObject = buildJsonObject { + put("username", username) + put("password", password) + put("remember_me", true) } - val body = RequestBody.create(MEDIA_TYPE, jsonObject.toString()) + val body = jsonObject.toString().toRequestBody(MEDIA_TYPE) val response = chain.proceed(POST("$baseUrl/api/login", headers, body)) if (response.code == 401) { throw Exception("Failed to login, check if username and password are correct") @@ -245,10 +249,9 @@ abstract class HentaiHand( if (response.body == null) throw Exception("Login response body is empty") try { - return JSONObject(response.body!!.string()) - .getJSONObject("auth") - .getString("access_token") - } catch (e: JSONException) { + // Returns access token as a string, unless unparseable + return json.parseToJsonElement(response.body!!.string()).jsonObject["auth"]!!.jsonObject["access-token"]!!.jsonPrimitive.content + } catch (e: IllegalArgumentException) { throw Exception("Cannot parse login response body") } } @@ -357,8 +360,7 @@ abstract class HentaiHand( ) companion object { - @SuppressLint("SimpleDateFormat") - private val DATE_FORMAT = SimpleDateFormat("yyyy-dd-MM") + private val DATE_FORMAT = SimpleDateFormat("yyyy-dd-MM", Locale.US) private val MEDIA_TYPE = "application/json; charset=utf-8".toMediaTypeOrNull() private const val USERNAME_TITLE = "Username" private const val USERNAME_DEFAULT = "" diff --git a/src/all/hentaihand/src/eu/kanade/tachiyomi/extension/all/hentaihand/HentaiHandFactory.kt b/src/all/hentaihand/src/eu/kanade/tachiyomi/extension/all/hentaihand/HentaiHandFactory.kt index 60e2415dc..b4caff519 100644 --- a/src/all/hentaihand/src/eu/kanade/tachiyomi/extension/all/hentaihand/HentaiHandFactory.kt +++ b/src/all/hentaihand/src/eu/kanade/tachiyomi/extension/all/hentaihand/HentaiHandFactory.kt @@ -50,7 +50,7 @@ class HentaiHandFactory : SourceFactory { ) } -class HentaiHandOther : HentaiHand("other", extraName = " (Unfiltered)") +class HentaiHandOther : HentaiHand("all", extraName = " (Unfiltered)") class HentaiHandEn : HentaiHand("en", 1) class HentaiHandZh : HentaiHand("zh", 2) class HentaiHandJa : HentaiHand("ja", 3) diff --git a/src/all/simplyhentai/build.gradle b/src/all/simplyhentai/build.gradle index c2aa76ceb..01d018ebc 100644 --- a/src/all/simplyhentai/build.gradle +++ b/src/all/simplyhentai/build.gradle @@ -1,11 +1,12 @@ apply plugin: 'com.android.application' apply plugin: 'kotlin-android' +apply plugin: 'kotlinx-serialization' ext { extName = 'Simply Hentai' pkgNameSuffix = 'all.simplyhentai' extClass = '.SimplyHentaiFactory' - extVersionCode = 3 + extVersionCode = 4 libVersion = '1.2' containsNsfw = true } diff --git a/src/all/simplyhentai/src/eu/kanade/tachiyomi/extension/all/simplyhentai/SimplyHentai.kt b/src/all/simplyhentai/src/eu/kanade/tachiyomi/extension/all/simplyhentai/SimplyHentai.kt index 64a2a90c8..51e184b92 100644 --- a/src/all/simplyhentai/src/eu/kanade/tachiyomi/extension/all/simplyhentai/SimplyHentai.kt +++ b/src/all/simplyhentai/src/eu/kanade/tachiyomi/extension/all/simplyhentai/SimplyHentai.kt @@ -1,12 +1,6 @@ package eu.kanade.tachiyomi.extension.all.simplyhentai import android.annotation.SuppressLint -import com.github.salomonbrys.kotson.forEach -import com.github.salomonbrys.kotson.fromJson -import com.github.salomonbrys.kotson.get -import com.github.salomonbrys.kotson.string -import com.google.gson.Gson -import com.google.gson.JsonObject import eu.kanade.tachiyomi.network.GET import eu.kanade.tachiyomi.source.model.Filter import eu.kanade.tachiyomi.source.model.FilterList @@ -15,12 +9,16 @@ import eu.kanade.tachiyomi.source.model.SChapter import eu.kanade.tachiyomi.source.model.SManga import eu.kanade.tachiyomi.source.online.ParsedHttpSource import eu.kanade.tachiyomi.util.asJsoup +import kotlinx.serialization.json.Json +import kotlinx.serialization.json.jsonObject +import kotlinx.serialization.json.jsonPrimitive import okhttp3.HttpUrl.Companion.toHttpUrlOrNull import okhttp3.OkHttpClient import okhttp3.Request import okhttp3.Response import org.jsoup.nodes.Document import org.jsoup.nodes.Element +import uy.kohesive.injekt.injectLazy import java.text.SimpleDateFormat abstract class SimplyHentai( @@ -36,7 +34,7 @@ abstract class SimplyHentai( override val client: OkHttpClient = network.cloudflareClient - private val gson = Gson() + private val json: Json by injectLazy() // Popular @@ -158,8 +156,13 @@ abstract class SimplyHentai( override fun pageListParse(response: Response): List { val pages = mutableListOf() - gson.fromJson(response.body!!.string()).forEach { _, jsonElement -> - pages.add(Page(pages.size, "", jsonElement["sizes"]["full"].string)) + json.parseToJsonElement(response.body!!.string()).jsonObject.forEach { + pages.add( + Page( + pages.size, "", + it.value.jsonObject["sizes"]!!.jsonObject["full"]!!.jsonPrimitive.content + ) + ) } return pages diff --git a/src/fr/kangaryu/build.gradle b/src/fr/kangaryu/build.gradle index 157994a71..92d8b4832 100644 --- a/src/fr/kangaryu/build.gradle +++ b/src/fr/kangaryu/build.gradle @@ -1,11 +1,12 @@ apply plugin: 'com.android.application' apply plugin: 'kotlin-android' +apply plugin: 'kotlinx-serialization' ext { extName = 'Kangaryu' pkgNameSuffix = 'fr.kangaryu' extClass = '.Kangaryu' - extVersionCode = 2 + extVersionCode = 3 libVersion = '1.2' } diff --git a/src/fr/kangaryu/src/eu/kanade/tachiyomi/extension/fr/kangaryu/Kangaryu.kt b/src/fr/kangaryu/src/eu/kanade/tachiyomi/extension/fr/kangaryu/Kangaryu.kt index 002435cee..3465a342d 100644 --- a/src/fr/kangaryu/src/eu/kanade/tachiyomi/extension/fr/kangaryu/Kangaryu.kt +++ b/src/fr/kangaryu/src/eu/kanade/tachiyomi/extension/fr/kangaryu/Kangaryu.kt @@ -1,11 +1,5 @@ package eu.kanade.tachiyomi.extension.fr.kangaryu -import com.github.salomonbrys.kotson.array -import com.github.salomonbrys.kotson.fromJson -import com.github.salomonbrys.kotson.get -import com.github.salomonbrys.kotson.string -import com.google.gson.Gson -import com.google.gson.JsonObject import eu.kanade.tachiyomi.network.GET import eu.kanade.tachiyomi.source.model.FilterList import eu.kanade.tachiyomi.source.model.MangasPage @@ -13,11 +7,16 @@ import eu.kanade.tachiyomi.source.model.Page import eu.kanade.tachiyomi.source.model.SChapter import eu.kanade.tachiyomi.source.model.SManga import eu.kanade.tachiyomi.source.online.ParsedHttpSource +import kotlinx.serialization.json.Json +import kotlinx.serialization.json.jsonArray +import kotlinx.serialization.json.jsonObject +import kotlinx.serialization.json.jsonPrimitive import okhttp3.OkHttpClient import okhttp3.Request import okhttp3.Response import org.jsoup.nodes.Document import org.jsoup.nodes.Element +import uy.kohesive.injekt.injectLazy import java.text.SimpleDateFormat import java.util.Locale @@ -87,14 +86,15 @@ class Kangaryu : ParsedHttpSource() { return GET("$baseUrl/search?query=$query", headers) } - private val gson by lazy { Gson() } + private val json: Json by injectLazy() override fun searchMangaParse(response: Response): MangasPage { - val mangas = gson.fromJson(response.body!!.string())["suggestions"].array.map { json -> + val mangas = json.parseToJsonElement(response.body!!.string()).jsonObject["suggestions"]!!.jsonArray.map { + val data = it.jsonObject["data"]!!.jsonPrimitive.content SManga.create().apply { - url = "/manga/${json["data"].string}" - title = json["value"].string - thumbnail_url = "https://kangaryu-team.fr/uploads/manga/${json["data"].string}/cover/cover_250x350.jpg" + url = "/manga/$data" + title = it.jsonObject["value"]!!.jsonPrimitive.content + thumbnail_url = "https://kangaryu-team.fr/uploads/manga/$data/cover/cover_250x350.jpg" } } return MangasPage(mangas, false)