mirror of
https://github.com/kodjodevf/mangayomi-extensions.git
synced 2026-02-14 02:41:39 +00:00
608 lines
20 KiB
JavaScript
608 lines
20 KiB
JavaScript
const mangayomiSources = [{
|
|
"name": "Aniplay",
|
|
"lang": "en",
|
|
"baseUrl": "https://aniplaynow.live",
|
|
"apiUrl": "https://aniplaynow.live",
|
|
"iconUrl": "https://www.google.com/s2/favicons?sz=128&domain=https://aniplaynow.live/",
|
|
"typeSource": "single",
|
|
"itemType": 1,
|
|
"version": "1.2.2",
|
|
"dateFormat": "",
|
|
"dateFormatLocale": "",
|
|
"pkgPath": "anime/src/en/aniplay.js"
|
|
}];
|
|
|
|
class DefaultExtension extends MProvider {
|
|
|
|
constructor() {
|
|
super();
|
|
this.client = new Client();
|
|
}
|
|
|
|
getPreference(key) {
|
|
const preferences = new SharedPreferences();
|
|
return preferences.get(key);
|
|
}
|
|
|
|
|
|
// code from torrentioanime.js
|
|
anilistQuery() {
|
|
return `
|
|
query ($page: Int, $perPage: Int, $sort: [MediaSort], $search: String) {
|
|
Page(page: $page, perPage: $perPage) {
|
|
pageInfo {
|
|
currentPage
|
|
hasNextPage
|
|
}
|
|
media(type: ANIME, sort: $sort, search: $search, status_in: [RELEASING, FINISHED, NOT_YET_RELEASED]) {
|
|
id
|
|
title {
|
|
romaji
|
|
english
|
|
native
|
|
}
|
|
coverImage {
|
|
extraLarge
|
|
large
|
|
}
|
|
description
|
|
status
|
|
tags {
|
|
name
|
|
}
|
|
genres
|
|
studios {
|
|
nodes {
|
|
name
|
|
}
|
|
}
|
|
countryOfOrigin
|
|
isAdult
|
|
}
|
|
}
|
|
}
|
|
`.trim();
|
|
}
|
|
|
|
// code from torrentioanime.js
|
|
anilistLatestQuery() {
|
|
const currentTimeInSeconds = Math.floor(Date.now() / 1000);
|
|
return `
|
|
query ($page: Int, $perPage: Int, $sort: [AiringSort]) {
|
|
Page(page: $page, perPage: $perPage) {
|
|
pageInfo {
|
|
currentPage
|
|
hasNextPage
|
|
}
|
|
airingSchedules(
|
|
airingAt_greater: 0
|
|
airingAt_lesser: ${currentTimeInSeconds - 10000}
|
|
sort: $sort
|
|
) {
|
|
media {
|
|
id
|
|
title {
|
|
romaji
|
|
english
|
|
native
|
|
}
|
|
coverImage {
|
|
extraLarge
|
|
large
|
|
}
|
|
description
|
|
status
|
|
tags {
|
|
name
|
|
}
|
|
genres
|
|
studios {
|
|
nodes {
|
|
name
|
|
}
|
|
}
|
|
countryOfOrigin
|
|
isAdult
|
|
}
|
|
}
|
|
}
|
|
}
|
|
`.trim();
|
|
}
|
|
|
|
// code from torrentioanime.js
|
|
async getAnimeDetails(anilistId) {
|
|
const query = `
|
|
query($id: Int){
|
|
Media(id: $id){
|
|
id
|
|
title {
|
|
romaji
|
|
english
|
|
native
|
|
}
|
|
coverImage {
|
|
extraLarge
|
|
large
|
|
}
|
|
description
|
|
status
|
|
tags {
|
|
name
|
|
}
|
|
genres
|
|
studios {
|
|
nodes {
|
|
name
|
|
}
|
|
}
|
|
format
|
|
countryOfOrigin
|
|
isAdult
|
|
}
|
|
}
|
|
`.trim();
|
|
|
|
const variables = JSON.stringify({ id: anilistId });
|
|
|
|
const res = await this.makeGraphQLRequest(query, variables);
|
|
const media = JSON.parse(res.body).data.Media;
|
|
const anime = {};
|
|
anime.name = (() => {
|
|
var preferenceTitle = this.getPreference("aniplay_pref_title");
|
|
switch (preferenceTitle) {
|
|
case "romaji":
|
|
return media?.title?.romaji || "";
|
|
case "english":
|
|
return media?.title?.english?.trim() || media?.title?.romaji || "";
|
|
case "native":
|
|
return media?.title?.native || "";
|
|
default:
|
|
return "";
|
|
}
|
|
})();
|
|
anime.imageUrl = media?.coverImage?.extraLarge ||
|
|
media?.coverImage?.large || "";
|
|
anime.description = (media?.description || "No Description")
|
|
.replace(/<br><br>/g, "\n")
|
|
.replace(/<.*?>/g, "");
|
|
|
|
anime.status = (() => {
|
|
switch (media?.status) {
|
|
case "RELEASING":
|
|
return 0;
|
|
case "FINISHED":
|
|
return 1;
|
|
case "HIATUS":
|
|
return 2;
|
|
case "NOT_YET_RELEASED":
|
|
return 3;
|
|
default:
|
|
return 5;
|
|
}
|
|
})();
|
|
|
|
const tagsList = media?.tags?.map(tag => tag.name).filter(Boolean) || [];
|
|
const genresList = media?.genres || [];
|
|
anime.genre = [...new Set([...tagsList, ...genresList])].sort();
|
|
const studiosList = media?.studios?.nodes?.map(node => node.name).filter(Boolean) || [];
|
|
anime.author = studiosList.sort().join(", ");
|
|
anime.format = media.format
|
|
return anime;
|
|
}
|
|
|
|
// code from torrentioanime.js
|
|
async makeGraphQLRequest(query, variables) {
|
|
const res = await this.client.post("https://graphql.anilist.co", {},
|
|
{
|
|
query, variables
|
|
});
|
|
return res;
|
|
}
|
|
|
|
// code from torrentioanime.js
|
|
parseSearchJson(jsonLine, isLatestQuery = false) {
|
|
const jsonData = JSON.parse(jsonLine);
|
|
jsonData.type = isLatestQuery ? "AnilistMetaLatest" : "AnilistMeta";
|
|
const metaData = jsonData;
|
|
|
|
const mediaList = metaData.type == "AnilistMeta"
|
|
? metaData.data?.Page?.media || []
|
|
: metaData.data?.Page?.airingSchedules.map(schedule => schedule.media) || [];
|
|
|
|
const hasNextPage = metaData.type == "AnilistMeta" || metaData.type == "AnilistMetaLatest"
|
|
? metaData.data?.Page?.pageInfo?.hasNextPage || false
|
|
: false;
|
|
|
|
const animeList = mediaList
|
|
.filter(media => !((media?.countryOfOrigin === "CN" || media?.isAdult) && isLatestQuery))
|
|
.map(media => {
|
|
const anime = {};
|
|
anime.link = media?.id?.toString() || "";
|
|
anime.name = (() => {
|
|
var preferenceTitle = this.getPreference("aniplay_pref_title");
|
|
switch (preferenceTitle) {
|
|
case "romaji":
|
|
return media?.title?.romaji || "";
|
|
case "english":
|
|
return media?.title?.english?.trim() || media?.title?.romaji || "";
|
|
case "native":
|
|
return media?.title?.native || "";
|
|
default:
|
|
return "";
|
|
}
|
|
})();
|
|
anime.imageUrl = media?.coverImage?.extraLarge ||
|
|
media?.coverImage?.large || "";
|
|
|
|
return anime;
|
|
});
|
|
|
|
return { "list": animeList, "hasNextPage": hasNextPage };
|
|
}
|
|
|
|
async getPopular(page) {
|
|
const variables = JSON.stringify({
|
|
page: page,
|
|
perPage: 30,
|
|
sort: "TRENDING_DESC"
|
|
});
|
|
|
|
const res = await this.makeGraphQLRequest(this.anilistQuery(), variables);
|
|
|
|
return this.parseSearchJson(res.body)
|
|
}
|
|
|
|
async getLatestUpdates(page) {
|
|
const variables = JSON.stringify({
|
|
page: page,
|
|
perPage: 30,
|
|
sort: "TIME_DESC"
|
|
});
|
|
|
|
const res = await this.makeGraphQLRequest(this.anilistLatestQuery(), variables);
|
|
|
|
return this.parseSearchJson(res.body, true)
|
|
}
|
|
|
|
async search(query, page, filters) {
|
|
const variables = JSON.stringify({
|
|
page: page,
|
|
perPage: 30,
|
|
sort: "POPULARITY_DESC",
|
|
search: query
|
|
});
|
|
|
|
const res = await this.makeGraphQLRequest(this.anilistQuery(), variables);
|
|
|
|
return this.parseSearchJson(res.body)
|
|
}
|
|
|
|
get supportsLatest() {
|
|
throw new Error("supportsLatest not implemented");
|
|
}
|
|
|
|
async aniplayRequest(slug, body) {
|
|
var next_action = ""
|
|
|
|
if (slug.indexOf("info/") > -1) {
|
|
next_action = 'f3422af67c84852f5e63d50e1f51718f1c0225c4'
|
|
} else if (slug.indexOf("watch/") > -1) {
|
|
next_action = '5dbcd21c7c276c4d15f8de29d9ef27aef5ea4a5e'
|
|
}
|
|
var baseUrl = "https://" + this.getPreference("aniplay_override_base_url")
|
|
var url = `${baseUrl}/anime/${slug}`
|
|
var headers = {
|
|
"referer": baseUrl,
|
|
'next-action': next_action,
|
|
"Content-Type": "application/json",
|
|
}
|
|
|
|
var response = await new Client().post(url, headers, body);
|
|
|
|
if (response.statusCode != 200) {
|
|
throw new Error("Error: " + response.statusText);
|
|
}
|
|
return JSON.parse(response.body.split('1:')[1])
|
|
|
|
}
|
|
|
|
async getDetail(url) {
|
|
var anilistId = url
|
|
var animeData = await this.getAnimeDetails(anilistId)
|
|
|
|
|
|
var slug = `info/${anilistId}`
|
|
var body = [anilistId, true, false]
|
|
var result = await this.aniplayRequest(slug, body)
|
|
if (result.length < 1) {
|
|
throw new Error("Error: No data found for the given URL");
|
|
}
|
|
|
|
var user_provider = this.getPreference("aniplay_pref_provider_3");
|
|
var choices = result
|
|
if (user_provider !== "all") {
|
|
for (var ch of result) {
|
|
if (ch["providerId"] == user_provider) {
|
|
choices = [ch]
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
for (const choice of choices) {
|
|
var user_mark_filler_ep = this.getPreference("aniplay_pref_mark_filler");
|
|
var chapters = []
|
|
var epList = choice.episodes
|
|
for (var ep of epList) {
|
|
var title = ep.title
|
|
var num = ep.number
|
|
var isFiller = ep.isFiller
|
|
|
|
var scanlator = isFiller && user_mark_filler_ep ? `Filler` : null;
|
|
|
|
var dateUpload = "createdAt" in ep ? new Date(ep.createdAt) : Date.now()
|
|
dateUpload = dateUpload.valueOf().toString();
|
|
delete ep.img
|
|
delete ep.title
|
|
delete ep.description
|
|
delete ep.isFiller
|
|
var epUrl = `${anilistId}||${JSON.stringify(ep)}||${choice.providerId}`
|
|
chapters.push({ name: `E${num}: ${title}`, url: epUrl, dateUpload, scanlator })
|
|
}
|
|
}
|
|
|
|
|
|
var format = animeData.format
|
|
if (format === "MOVIE") chapters[0].name = "Movie"
|
|
|
|
var baseUrl = "https://" + this.getPreference("aniplay_override_base_url")
|
|
animeData.link = `${baseUrl}/anime/${slug}`
|
|
animeData.chapters = chapters.reverse()
|
|
return animeData
|
|
}
|
|
|
|
|
|
// Sorts streams based on user preference.
|
|
async sortStreams(streams) {
|
|
var sortedStreams = [];
|
|
var copyStreams = streams.slice()
|
|
|
|
var pref = await this.getPreference("aniplay_pref_video_resolution");
|
|
for (var stream of streams) {
|
|
|
|
if (stream.quality.indexOf(pref) > -1) {
|
|
sortedStreams.push(stream);
|
|
var index = copyStreams.indexOf(stream);
|
|
if (index > -1) {
|
|
copyStreams.splice(index, 1);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
return [...sortedStreams, ...copyStreams]
|
|
}
|
|
|
|
// Adds proxy to streams
|
|
async streamProxy(providerId, streams) {
|
|
var proxyBaseUrl = this.getPreference("aniplay_stream_proxy");
|
|
var slug = "/fetch?url="
|
|
var ref = "&ref="
|
|
if (providerId == "yuki") {
|
|
slug = "/yukiprox?url="
|
|
ref = ""
|
|
} else if (providerId == "pahe") {
|
|
ref += "https://kwik.si"
|
|
} else if (providerId === "maze") {
|
|
ref += "https://animez.org"
|
|
} else if (providerId === "kuro") {
|
|
ref = ""
|
|
}
|
|
|
|
streams.forEach(stream => {
|
|
stream.url = proxyBaseUrl + slug + stream.url + ref;
|
|
stream.originalUrl = proxyBaseUrl + slug + stream.originalUrl + ref;
|
|
});
|
|
|
|
return streams
|
|
}
|
|
|
|
// Extracts the streams url for different resolutions from a hls stream.
|
|
async extractStreams(url, providerId, hdr = {}) {
|
|
const response = await new Client().get(url, hdr);
|
|
const body = response.body;
|
|
const lines = body.split('\n');
|
|
var streams = [{
|
|
url: url,
|
|
originalUrl: url,
|
|
quality: `Auto - ${providerId}`,
|
|
headers: hdr
|
|
}];
|
|
// Pahe only has auto
|
|
if (providerId === "pahe") {
|
|
return streams;
|
|
}
|
|
for (let i = 0; i < lines.length; i++) {
|
|
if (lines[i].startsWith('#EXT-X-STREAM-INF:')) {
|
|
var resolution = lines[i].match(/RESOLUTION=(\d+x\d+)/)[1];
|
|
var m3u8Url = lines[i + 1].trim();
|
|
if (providerId === "yuki") {
|
|
var orginalUrl = url
|
|
m3u8Url = orginalUrl.replace("master.m3u8", m3u8Url)
|
|
}
|
|
streams.push({
|
|
url: m3u8Url,
|
|
originalUrl: m3u8Url,
|
|
quality: `${resolution} - ${providerId}`,
|
|
headers: hdr
|
|
});
|
|
}
|
|
}
|
|
return streams
|
|
|
|
}
|
|
|
|
async getAnyStreams(result) {
|
|
var m3u8Url = result.sources[0].url
|
|
return await this.extractStreams(m3u8Url, "any");
|
|
}
|
|
|
|
async getYukiStreams(result) {
|
|
var m3u8Url = result.sources[0].url
|
|
var streams = await this.extractStreams(m3u8Url, "yuki");
|
|
|
|
var subtitles = []
|
|
result.subtitles.forEach(sub => {
|
|
var label = sub.label
|
|
if (label.indexOf("thumbnail") < 0) { // thumbnails shouldnt be included
|
|
subtitles.push({
|
|
"label": label,
|
|
"file": sub.url,
|
|
});
|
|
}
|
|
})
|
|
streams[0].subtitles = subtitles
|
|
|
|
return streams
|
|
}
|
|
|
|
async getPaheStreams(result) {
|
|
var m3u8Url = result.sources[0].url
|
|
var hdr = result.headers;
|
|
return await this.extractStreams(m3u8Url, "pahe", hdr);
|
|
}
|
|
|
|
async getMazeStreams(result) {
|
|
var m3u8Url = result.sources[0].url
|
|
var hdr = result.headers;
|
|
return await this.extractStreams(m3u8Url, "maze", hdr);
|
|
}
|
|
|
|
async getKuroStreams(result) {
|
|
var links = result.sources
|
|
var hdr = result.headers;
|
|
var streams = [];
|
|
for (var stream of links) {
|
|
var quality = stream.quality
|
|
quality = quality == "default" ? "auto" : quality
|
|
streams.push({
|
|
url: stream.url,
|
|
originalUrl: stream.url,
|
|
quality: `${quality} - kuro`,
|
|
headers: hdr
|
|
});
|
|
}
|
|
return streams
|
|
}
|
|
|
|
// For anime episode video list
|
|
async getVideoList(url) {
|
|
var urlSplits = url.split("||")
|
|
var anilistId = urlSplits[0]
|
|
var epData = JSON.parse(urlSplits[1])
|
|
var providerId = urlSplits[2]
|
|
|
|
var user_audio_type = this.getPreference("aniplay_pref_audio_type");
|
|
var subOrDub = epData.hasDub && user_audio_type === "dub" ? "dub" : "sub"
|
|
|
|
var slug = `watch/${anilistId}`
|
|
var body = [
|
|
anilistId,
|
|
providerId,
|
|
epData.id,
|
|
epData.number.toString(),
|
|
subOrDub
|
|
]
|
|
var result = await this.aniplayRequest(slug, body)
|
|
if (result === null) {
|
|
throw new Error("Error: No data found for the given URL");
|
|
}
|
|
|
|
var streams = []
|
|
if (providerId == "yuki") {
|
|
streams = await this.getYukiStreams(result)
|
|
} else if (providerId == "pahe") {
|
|
streams = await this.getPaheStreams(result)
|
|
} else if (providerId === "maze") {
|
|
streams = await this.getMazeStreams(result)
|
|
} else if (providerId === "kuro") {
|
|
streams = await this.getKuroStreams(result)
|
|
} else {
|
|
streams = await this.getAnyStreams(result) //If new provider found getAnyStreams will extract only the streams
|
|
}
|
|
|
|
streams = await this.streamProxy(providerId, streams)
|
|
return await this.sortStreams(streams)
|
|
}
|
|
|
|
getSourcePreferences() {
|
|
return [{
|
|
key: 'aniplay_override_base_url',
|
|
listPreference: {
|
|
title: 'Override base url',
|
|
summary: '',
|
|
valueIndex: 0,
|
|
entries: ["aniplaynow.live (Main)", "aniplay.lol (Backup)"],
|
|
entryValues: ["aniplaynow.live", "aniplay.lol"]
|
|
}
|
|
},
|
|
{
|
|
"key": "aniplay_pref_title",
|
|
"listPreference": {
|
|
"title": "Preferred Title",
|
|
"summary": "",
|
|
"valueIndex": 0,
|
|
"entries": ["Romaji", "English", "Native"],
|
|
"entryValues": ["romaji", "english", "native"],
|
|
}
|
|
},
|
|
{
|
|
"key": "aniplay_pref_provider_3",
|
|
"listPreference": {
|
|
"title": "Preferred provider",
|
|
"summary": "",
|
|
"valueIndex": 0,
|
|
"entries": ["Any", "Yuki", "Pahe", "Maze", "Kuro"],
|
|
"entryValues": ["any", "yuki", "pahe", "maze", "kuro"],
|
|
}
|
|
}, {
|
|
"key": "aniplay_pref_mark_filler",
|
|
"switchPreferenceCompat": {
|
|
"title": "Mark filler episodes",
|
|
"summary": "Filler episodes will be marked with (F)",
|
|
"value": false
|
|
}
|
|
},
|
|
{
|
|
"key": "aniplay_pref_audio_type",
|
|
"listPreference": {
|
|
"title": "Preferred audio type",
|
|
"summary": "Sub/Dub",
|
|
"valueIndex": 0,
|
|
"entries": ["Sub", "Dub"],
|
|
"entryValues": ["sub", "dub"],
|
|
}
|
|
}, {
|
|
key: 'aniplay_pref_video_resolution',
|
|
listPreference: {
|
|
title: 'Preferred video resolution',
|
|
summary: '',
|
|
valueIndex: 0,
|
|
entries: ["Auto", "1080p", "720p", "480p", "360p"],
|
|
entryValues: ["auto", "1080", "720", "480", "360"]
|
|
}
|
|
}, {
|
|
key: "aniplay_stream_proxy",
|
|
editTextPreference: {
|
|
title: "Override stream proxy url",
|
|
summary: "https://prox.aniplaynow.live",
|
|
value: "https://prox.aniplaynow.live",
|
|
dialogTitle: "Override stream proxy url",
|
|
dialogMessage: "",
|
|
}
|
|
}
|
|
|
|
]
|
|
}
|
|
}
|