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.3.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(/

/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 baseUrl = "https://" + this.getPreference("aniplay_override_base_url") var next_action_overrides = this.getPreference("aniplay_next_action_key").split("||") var next_action_key = 0 if (baseUrl.endsWith(".lol")) { next_action_key = 1 } var next_action_values = next_action_overrides[next_action_key].split(":") var next_action = "" if (slug.indexOf("info/") > -1) { next_action = next_action_values[0] } else if (slug.indexOf("watch/") > -1) { next_action = next_action_values[1] } 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") { ref += "https://hianime.to" } else if (providerId == "pahe") { ref += "https://kwik.si" } else if (providerId === "maze") { ref += "" } 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 = {}) { var streams = [{ url: url, originalUrl: url, quality: `Auto - ${providerId}`, headers: hdr }]; var doExtract = this.getPreference("aniplay_pref_extract_streams"); // Pahe only has auto if (providerId === "pahe" || !doExtract) { return streams; } const response = await new Client().get(url, hdr); const body = response.body; const lines = body.split('\n'); 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 } if (this.getPreference("aniplay_proxy")) 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_next_action_key", editTextPreference: { title: "Override next_action key", summary: "", value: "7f328d44382d74f2942c42d0bc9915b2d510628a02:7f702090c0d779331a0b55e1ee0cfea85ff4cb963a||7fe26ef575b85a8a7166a411dba3a21263fe5b9306:7fbb025dd45b14410eb1a8ffcbb4615b0a5a1e5c3c", dialogTitle: "Override next_action key", dialogMessage: "", } }, { "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_extract_streams", "switchPreferenceCompat": { 'title': 'Split stream into different quality streams', "summary": "Split stream Auto into 360p/720p/1080p", "value": true } }, { 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_proxy", switchPreferenceCompat: { title: 'Enable stream proxy', summary: "", value: false } }, { 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: "", } } ] } }