Remove Pufei Manhua: site is down (#12826)
This commit is contained in:
parent
4a54a8c801
commit
5975d2528e
|
@ -1,2 +0,0 @@
|
|||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<manifest package="eu.kanade.tachiyomi.extension" />
|
|
@ -1,15 +0,0 @@
|
|||
apply plugin: 'com.android.application'
|
||||
apply plugin: 'kotlin-android'
|
||||
|
||||
ext {
|
||||
extName = 'Pufei Manhua'
|
||||
pkgNameSuffix = 'zh.pufei'
|
||||
extClass = '.Pufei'
|
||||
extVersionCode = 10
|
||||
}
|
||||
|
||||
apply from: "$rootDir/common.gradle"
|
||||
|
||||
dependencies {
|
||||
implementation 'com.github.stevenyomi:unpacker:12a09e3c1a' // 1.1
|
||||
}
|
Binary file not shown.
Before Width: | Height: | Size: 5.2 KiB |
Binary file not shown.
Before Width: | Height: | Size: 3.1 KiB |
Binary file not shown.
Before Width: | Height: | Size: 7.4 KiB |
Binary file not shown.
Before Width: | Height: | Size: 12 KiB |
Binary file not shown.
Before Width: | Height: | Size: 21 KiB |
Binary file not shown.
Before Width: | Height: | Size: 44 KiB |
|
@ -1,58 +0,0 @@
|
|||
package eu.kanade.tachiyomi.extension.zh.pufei
|
||||
|
||||
import android.os.SystemClock
|
||||
import okhttp3.Interceptor
|
||||
import okhttp3.Response
|
||||
import java.io.IOException
|
||||
import java.util.concurrent.TimeUnit
|
||||
|
||||
// See https://github.com/tachiyomiorg/tachiyomi/pull/7389
|
||||
internal class NonblockingRateLimiter(
|
||||
private val permits: Int,
|
||||
period: Long = 1,
|
||||
unit: TimeUnit = TimeUnit.SECONDS,
|
||||
) : Interceptor {
|
||||
|
||||
private val requestQueue = ArrayList<Long>(permits)
|
||||
private val rateLimitMillis = unit.toMillis(period)
|
||||
|
||||
override fun intercept(chain: Interceptor.Chain): Response {
|
||||
// Ignore canceled calls, otherwise they would jam the queue
|
||||
if (chain.call().isCanceled()) {
|
||||
throw IOException()
|
||||
}
|
||||
|
||||
synchronized(requestQueue) {
|
||||
val now = SystemClock.elapsedRealtime()
|
||||
val waitTime = if (requestQueue.size < permits) {
|
||||
0
|
||||
} else {
|
||||
val oldestReq = requestQueue[0]
|
||||
val newestReq = requestQueue[permits - 1]
|
||||
|
||||
if (newestReq - oldestReq > rateLimitMillis) {
|
||||
0
|
||||
} else {
|
||||
oldestReq + rateLimitMillis - now // Remaining time
|
||||
}
|
||||
}
|
||||
|
||||
// Final check
|
||||
if (chain.call().isCanceled()) {
|
||||
throw IOException()
|
||||
}
|
||||
|
||||
if (requestQueue.size == permits) {
|
||||
requestQueue.removeAt(0)
|
||||
}
|
||||
if (waitTime > 0) {
|
||||
requestQueue.add(now + waitTime)
|
||||
Thread.sleep(waitTime) // Sleep inside synchronized to pause queued requests
|
||||
} else {
|
||||
requestQueue.add(now)
|
||||
}
|
||||
}
|
||||
|
||||
return chain.proceed(chain.request())
|
||||
}
|
||||
}
|
|
@ -1,41 +0,0 @@
|
|||
package eu.kanade.tachiyomi.extension.zh.pufei
|
||||
|
||||
import eu.kanade.tachiyomi.network.GET
|
||||
import okhttp3.Interceptor
|
||||
import okhttp3.MediaType.Companion.toMediaType
|
||||
import okhttp3.Response
|
||||
import okhttp3.ResponseBody.Companion.asResponseBody
|
||||
|
||||
object OctetStreamInterceptor : Interceptor {
|
||||
|
||||
override fun intercept(chain: Interceptor.Chain): Response {
|
||||
val request = chain.request()
|
||||
val response = chain.proceed(request)
|
||||
|
||||
if (response.header("Content-Type") != "application/octet-stream") {
|
||||
return response
|
||||
}
|
||||
|
||||
if (response.header("Content-Length")!!.toInt() < 100) { // usually 96
|
||||
// The actual URL is '/.../xxx.jpg/0'.
|
||||
val peek = response.peekBody(100).string()
|
||||
if (peek.startsWith("The actual URL")) {
|
||||
response.body!!.close()
|
||||
val actualPath = peek.substringAfter('\'').substringBeforeLast('\'')
|
||||
return chain.proceed(GET("https://manhua.acimg.cn$actualPath"))
|
||||
}
|
||||
}
|
||||
|
||||
val url = request.url.encodedPath
|
||||
val mediaType = when {
|
||||
url.endsWith(".h") -> webpMediaType
|
||||
url.contains(".jpg") -> jpegMediaType
|
||||
else -> return response
|
||||
}
|
||||
val body = response.body!!.source().asResponseBody(mediaType)
|
||||
return response.newBuilder().body(body).build()
|
||||
}
|
||||
|
||||
private val jpegMediaType = "image/jpeg".toMediaType()
|
||||
private val webpMediaType = "image/webp".toMediaType()
|
||||
}
|
|
@ -1,223 +0,0 @@
|
|||
package eu.kanade.tachiyomi.extension.zh.pufei
|
||||
|
||||
import android.app.Application
|
||||
import android.content.SharedPreferences
|
||||
import android.util.Base64
|
||||
import androidx.preference.ListPreference
|
||||
import androidx.preference.PreferenceScreen
|
||||
import com.github.stevenyomi.unpacker.ProgressiveParser
|
||||
import com.github.stevenyomi.unpacker.Unpacker
|
||||
import eu.kanade.tachiyomi.network.GET
|
||||
import eu.kanade.tachiyomi.network.POST
|
||||
import eu.kanade.tachiyomi.network.asObservableSuccess
|
||||
import eu.kanade.tachiyomi.source.ConfigurableSource
|
||||
import eu.kanade.tachiyomi.source.model.FilterList
|
||||
import eu.kanade.tachiyomi.source.model.MangasPage
|
||||
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 okhttp3.FormBody
|
||||
import okhttp3.Request
|
||||
import okhttp3.Response
|
||||
import org.jsoup.nodes.Document
|
||||
import org.jsoup.nodes.Element
|
||||
import org.jsoup.select.Evaluator
|
||||
import rx.Observable
|
||||
import uy.kohesive.injekt.Injekt
|
||||
import uy.kohesive.injekt.api.get
|
||||
|
||||
// Uses www733dm/IMH/dm456 theme
|
||||
class Pufei : ParsedHttpSource(), ConfigurableSource {
|
||||
|
||||
override val name = "扑飞漫画"
|
||||
override val lang = "zh"
|
||||
override val supportsLatest = true
|
||||
|
||||
private val preferences: SharedPreferences =
|
||||
Injekt.get<Application>().getSharedPreferences("source_$id", 0x0000)
|
||||
|
||||
private val domain = preferences.getString(MIRROR_PREF, "0")!!
|
||||
.toInt().coerceAtMost(MIRRORS.size - 1).let { MIRRORS[it] }
|
||||
|
||||
override val baseUrl = "http://m.$domain"
|
||||
private val pcUrl = "http://www.$domain"
|
||||
|
||||
override val client = network.client.newBuilder()
|
||||
.addInterceptor(NonblockingRateLimiter(2))
|
||||
.addInterceptor(OctetStreamInterceptor)
|
||||
.build()
|
||||
|
||||
private val searchClient = network.client.newBuilder()
|
||||
.followRedirects(false)
|
||||
.build()
|
||||
|
||||
override fun popularMangaRequest(page: Int) = GET("$baseUrl/manhua/paihang.html", headers)
|
||||
override fun popularMangaNextPageSelector() = throw UnsupportedOperationException("Not used.")
|
||||
override fun popularMangaSelector() = "ul#detail > li > a"
|
||||
override fun popularMangaFromElement(element: Element) = SManga.create().apply {
|
||||
url = element.attr("href").removeSuffix("/index.html")
|
||||
title = element.selectFirst(Evaluator.Tag("h3")).text()
|
||||
thumbnail_url = element.selectFirst(Evaluator.Tag("img")).attr("data-src")
|
||||
}
|
||||
|
||||
override fun popularMangaParse(response: Response): MangasPage {
|
||||
val document = response.asPufeiJsoup()
|
||||
val mangas = document.select(popularMangaSelector()).map { popularMangaFromElement(it) }
|
||||
return MangasPage(mangas, false)
|
||||
}
|
||||
|
||||
override fun latestUpdatesRequest(page: Int) = GET("$baseUrl/manhua/update.html", headers)
|
||||
override fun latestUpdatesNextPageSelector() = throw UnsupportedOperationException("Not used.")
|
||||
override fun latestUpdatesSelector() = popularMangaSelector()
|
||||
override fun latestUpdatesFromElement(element: Element) = popularMangaFromElement(element)
|
||||
|
||||
override fun latestUpdatesParse(response: Response): MangasPage {
|
||||
val document = response.asPufeiJsoup()
|
||||
val mangas = document.select(latestUpdatesSelector()).map { latestUpdatesFromElement(it) }
|
||||
return MangasPage(mangas, false)
|
||||
}
|
||||
|
||||
private val searchCache = HashMap<String, String>()
|
||||
|
||||
override fun searchMangaRequest(page: Int, query: String, filters: FilterList): Request {
|
||||
return if (query.isNotBlank()) {
|
||||
val path = searchCache.getOrPut(query) {
|
||||
val formBody = FormBody.Builder(GB2312)
|
||||
.addEncoded("tempid", "4")
|
||||
.addEncoded("show", "title,player,playadmin,bieming,pinyin")
|
||||
.add("keyboard", query)
|
||||
.build()
|
||||
val request = POST("$baseUrl/e/search/index.php", headers, formBody)
|
||||
searchClient.newCall(request).execute().header("location")!!
|
||||
}
|
||||
val sortQuery = parseSearchSort(filters)
|
||||
GET("$baseUrl/e/search/$path$sortQuery&page=${page - 1}")
|
||||
} else {
|
||||
val path = parseFilters(page, filters)
|
||||
if (path.isEmpty())
|
||||
popularMangaRequest(page)
|
||||
else
|
||||
GET("$baseUrl$path", headers)
|
||||
}
|
||||
}
|
||||
|
||||
override fun searchMangaNextPageSelector() = throw UnsupportedOperationException("Not used.")
|
||||
override fun searchMangaSelector() = popularMangaSelector()
|
||||
override fun searchMangaFromElement(element: Element) = popularMangaFromElement(element)
|
||||
|
||||
override fun searchMangaParse(response: Response): MangasPage {
|
||||
val document = response.asPufeiJsoup()
|
||||
val mangas = document.select(searchMangaSelector()).map { searchMangaFromElement(it) }
|
||||
val hasNextPage = run {
|
||||
for (element in document.body().children().asReversed()) {
|
||||
if (element.tagName() == "a") return@run true
|
||||
else if (element.tagName() == "b") return@run false
|
||||
}
|
||||
false
|
||||
}
|
||||
return MangasPage(mangas, hasNextPage)
|
||||
}
|
||||
|
||||
override fun getFilterList() = getFilters()
|
||||
|
||||
// 让 WebView 显示移动端页面
|
||||
override fun mangaDetailsRequest(manga: SManga) = GET(baseUrl + manga.url, headers)
|
||||
|
||||
override fun fetchMangaDetails(manga: SManga): Observable<SManga> =
|
||||
client.newCall(GET(pcUrl + manga.urlWithCheck(), headers)).asObservableSuccess()
|
||||
.map { mangaDetailsParse(it.asPufeiJsoup()) }
|
||||
|
||||
override fun mangaDetailsParse(document: Document) = SManga.create().apply {
|
||||
val details = document.selectFirst(Evaluator.Class("detailInfo")).children()
|
||||
title = details[0].child(0).text() // div.titleInfo > h1
|
||||
val genreList = mutableListOf<String>()
|
||||
for (item in details[1].children()) { // ul > li
|
||||
when (item.child(0).text()) { // span
|
||||
"作者:" -> author = item.ownText()
|
||||
"类别:" -> item.ownText().let { if (it.isNotEmpty()) genreList.add(it) }
|
||||
"关键词:" -> item.ownText().let { if (it.isNotEmpty()) genreList.addAll(it.split(',')) }
|
||||
}
|
||||
}
|
||||
author = author ?: details[0].ownText().removePrefix("作者:")
|
||||
if (genreList.isEmpty()) {
|
||||
genreList.add(document.selectFirst(Evaluator.Class("position")).child(1).text())
|
||||
}
|
||||
genre = genreList.joinToString()
|
||||
description = document.selectFirst("div.introduction")?.text() ?: details[2].ownText()
|
||||
status = SManga.UNKNOWN // 所有漫画的标记都是连载,所以没有意义,参见 baseUrl/manhua/wanjie.html
|
||||
thumbnail_url = document.selectFirst("img.pic").attr("src")
|
||||
}
|
||||
|
||||
override fun chapterListRequest(manga: SManga) = GET(pcUrl + manga.urlWithCheck(), headers)
|
||||
|
||||
override fun chapterListSelector() = "div.plistBox ul > li > a"
|
||||
|
||||
override fun chapterFromElement(element: Element) = SChapter.create().apply {
|
||||
url = element.attr("href")
|
||||
name = element.attr("title")
|
||||
}
|
||||
|
||||
override fun chapterListParse(response: Response): List<SChapter> {
|
||||
val document = response.asPufeiJsoup()
|
||||
val list = document.select(chapterListSelector()).map { chapterFromElement(it) }
|
||||
if (isNewDateLogic && list.isNotEmpty()) {
|
||||
val date = document.selectFirst("li.twoCol:contains(更新时间)").text().removePrefix("更新时间:").trim()
|
||||
list[0].date_upload = dateFormat.parse(date)?.time ?: 0
|
||||
}
|
||||
return list
|
||||
}
|
||||
|
||||
override fun pageListRequest(chapter: SChapter) = GET(baseUrl + chapter.url, headers)
|
||||
|
||||
override fun pageListParse(response: Response): List<Page> {
|
||||
val html = String(response.body!!.bytes(), GB2312).let(::ProgressiveParser)
|
||||
val base64 = html.substringBetween("cp=\"", "\"")
|
||||
val script = String(Base64.decode(base64, Base64.DEFAULT))
|
||||
val result = Unpacker.unpack(script, "[", "]")
|
||||
.ifEmpty { return emptyList() }
|
||||
.replace("\\", "")
|
||||
.removeSurrounding("\"").split("\",\"")
|
||||
// baseUrl/skin/2014mh/view.js (imgserver), mobileUrl/skin/main.js (IMH.reader)
|
||||
return result.mapIndexed { i, image ->
|
||||
val imageUrl = if (image.startsWith("http")) image else IMAGE_SERVER + image
|
||||
Page(i, imageUrl = imageUrl)
|
||||
}
|
||||
}
|
||||
|
||||
override fun pageListParse(document: Document) = throw UnsupportedOperationException("Not used.")
|
||||
|
||||
override fun imageUrlParse(document: Document) = throw UnsupportedOperationException("Not used.")
|
||||
|
||||
override fun setupPreferenceScreen(screen: PreferenceScreen) {
|
||||
ListPreference(screen.context).apply {
|
||||
key = MIRROR_PREF
|
||||
title = "使用镜像网站"
|
||||
summary = "选择要使用的镜像网站,重启生效\n已选择:%s"
|
||||
entries = MIRRORS_DESCRIPTION
|
||||
entryValues = MIRROR_VALUES
|
||||
setDefaultValue("0")
|
||||
}.let { screen.addPreference(it) }
|
||||
}
|
||||
|
||||
companion object {
|
||||
private const val MIRROR_PREF = "MIRROR"
|
||||
private val MIRROR_VALUES = arrayOf("0", "1", "2", "3", "4")
|
||||
private val MIRRORS = arrayOf(
|
||||
"pufei.cc",
|
||||
"pfmh.net",
|
||||
"alimanhua.com",
|
||||
"8nfw.com",
|
||||
"pufei5.com",
|
||||
)
|
||||
private val MIRRORS_DESCRIPTION = arrayOf(
|
||||
"pufei.cc",
|
||||
"pfmh.net",
|
||||
"alimanhua.com (阿狸漫画)",
|
||||
"8nfw.com (风之动漫)",
|
||||
"pufei5.com (不推荐)",
|
||||
)
|
||||
|
||||
private const val IMAGE_SERVER = "http://res.img.tueqi.com/"
|
||||
}
|
||||
}
|
|
@ -1,51 +0,0 @@
|
|||
package eu.kanade.tachiyomi.extension.zh.pufei
|
||||
|
||||
import eu.kanade.tachiyomi.source.model.Filter
|
||||
import eu.kanade.tachiyomi.source.model.FilterList
|
||||
|
||||
internal fun getFilters() = FilterList(
|
||||
Filter.Header("排序只对文本搜索和分类筛选有效"),
|
||||
SortFilter(),
|
||||
Filter.Separator(),
|
||||
Filter.Header("以下筛选最多使用一个,使用文本搜索时将会忽略"),
|
||||
CategoryFilter(),
|
||||
AlphabetFilter(),
|
||||
)
|
||||
|
||||
internal fun parseSearchSort(filters: FilterList): String =
|
||||
filters.filterIsInstance<SortFilter>().firstOrNull()?.let { SORT_QUERIES[it.state] } ?: ""
|
||||
|
||||
internal fun parseFilters(page: Int, filters: FilterList): String {
|
||||
val pageStr = if (page == 1) "" else "_$page"
|
||||
var category = 0
|
||||
var categorySort = 0
|
||||
var alphabet = 0
|
||||
for (filter in filters) when (filter) {
|
||||
is SortFilter -> categorySort = filter.state
|
||||
is CategoryFilter -> category = filter.state
|
||||
is AlphabetFilter -> alphabet = filter.state
|
||||
else -> {}
|
||||
}
|
||||
return if (category > 0) {
|
||||
"/${CATEGORY_KEYS[category]}/${SORT_KEYS[categorySort]}$pageStr.html"
|
||||
} else if (alphabet > 0) {
|
||||
"/mh/${ALPHABET[alphabet].lowercase()}/index$pageStr.html"
|
||||
} else {
|
||||
""
|
||||
}
|
||||
}
|
||||
|
||||
internal class SortFilter : Filter.Select<String>("排序", SORT_NAMES)
|
||||
|
||||
private val SORT_NAMES = arrayOf("添加时间", "更新时间", "点击次数")
|
||||
private val SORT_KEYS = arrayOf("index", "update", "view")
|
||||
private val SORT_QUERIES = arrayOf("&orderby=newstime", "&orderby=lastdotime", "&orderby=onclick")
|
||||
|
||||
internal class CategoryFilter : Filter.Select<String>("分类", CATEGORY_NAMES)
|
||||
|
||||
private val CATEGORY_NAMES = arrayOf("全部", "少年热血", "少女爱情", "武侠格斗", "科幻魔幻", "竞技体育", "搞笑喜剧", "耽美人生", "侦探推理", "恐怖灵异")
|
||||
private val CATEGORY_KEYS = arrayOf("", "shaonianrexue", "shaonvaiqing", "wuxiagedou", "kehuan", "jingjitiyu", "gaoxiaoxiju", "danmeirensheng", "zhentantuili", "kongbulingyi")
|
||||
|
||||
internal class AlphabetFilter : Filter.Select<String>("字母", ALPHABET)
|
||||
|
||||
private val ALPHABET = arrayOf("全部", "A", "B", "C", "D", "E", "F", "G", "H", "I", "J", "K", "L", "M", "N", "O", "P", "Q", "R", "S", "T", "U", "V", "W", "X", "Y", "Z")
|
|
@ -1,28 +0,0 @@
|
|||
package eu.kanade.tachiyomi.extension.zh.pufei
|
||||
|
||||
import eu.kanade.tachiyomi.AppInfo
|
||||
import eu.kanade.tachiyomi.source.model.SManga
|
||||
import okhttp3.Response
|
||||
import org.jsoup.Jsoup
|
||||
import org.jsoup.nodes.Document
|
||||
import java.text.SimpleDateFormat
|
||||
import java.util.Locale
|
||||
|
||||
internal val GB2312 = charset("GB2312")
|
||||
|
||||
internal fun Response.asPufeiJsoup(): Document =
|
||||
Jsoup.parse(String(body!!.bytes(), GB2312), request.url.toString())
|
||||
|
||||
internal fun SManga.urlWithCheck(): String {
|
||||
val result = url
|
||||
if (result.endsWith("/index.html")) {
|
||||
throw Exception("作品地址格式过期,请迁移更新")
|
||||
}
|
||||
return result
|
||||
}
|
||||
|
||||
internal val isNewDateLogic = AppInfo.getVersionCode() >= 81
|
||||
|
||||
internal val dateFormat by lazy {
|
||||
SimpleDateFormat("yyyy-MM-dd HH:mm", Locale.ENGLISH)
|
||||
}
|
Loading…
Reference in New Issue