From f0c98f9d0abad362859c71e06b84a73cd9a724e1 Mon Sep 17 00:00:00 2001 From: himanshu8443 Date: Fri, 15 Aug 2025 15:57:36 +0530 Subject: [PATCH] feat: add a111477 provider --- dist/a111477/catalog.js | 1 + dist/a111477/episodes.js | 1 + dist/a111477/meta.js | 1 + dist/a111477/posts.js | 1 + dist/a111477/stream.js | 1 + manifest.json | 10 ++- providers/a111477/catalog.ts | 24 +++++++ providers/a111477/episodes.ts | 69 ++++++++++++++++++ providers/a111477/meta.ts | 103 +++++++++++++++++++++++++++ providers/a111477/posts.ts | 128 ++++++++++++++++++++++++++++++++++ providers/a111477/stream.ts | 38 ++++++++++ 11 files changed, 376 insertions(+), 1 deletion(-) create mode 100644 dist/a111477/catalog.js create mode 100644 dist/a111477/episodes.js create mode 100644 dist/a111477/meta.js create mode 100644 dist/a111477/posts.js create mode 100644 dist/a111477/stream.js create mode 100644 providers/a111477/catalog.ts create mode 100644 providers/a111477/episodes.ts create mode 100644 providers/a111477/meta.ts create mode 100644 providers/a111477/posts.ts create mode 100644 providers/a111477/stream.ts diff --git a/dist/a111477/catalog.js b/dist/a111477/catalog.js new file mode 100644 index 0000000..760c0e5 --- /dev/null +++ b/dist/a111477/catalog.js @@ -0,0 +1 @@ +"use strict";Object.defineProperty(exports,"__esModule",{value:!0}),exports.genres=exports.catalog=void 0,exports.catalog=[{title:"Movies",filter:"/movies/"},{title:"TV Shows",filter:"/tvs/"},{title:"K-Drama",filter:"/kdrama/"},{title:"Asian Drama",filter:"/asiandrama/"},{title:"Misc",filter:"/misc/"}],exports.genres=[]; \ No newline at end of file diff --git a/dist/a111477/episodes.js b/dist/a111477/episodes.js new file mode 100644 index 0000000..cdc7f17 --- /dev/null +++ b/dist/a111477/episodes.js @@ -0,0 +1 @@ +"use strict";var __awaiter=this&&this.__awaiter||function(thisArg,_arguments,P,generator){return new(P||(P=Promise))(function(resolve,reject){function fulfilled(value){try{step(generator.next(value))}catch(e){reject(e)}}function rejected(value){try{step(generator.throw(value))}catch(e){reject(e)}}function step(result){var value;result.done?resolve(result.value):(value=result.value,value instanceof P?value:new P(function(resolve){resolve(value)})).then(fulfilled,rejected)}step((generator=generator.apply(thisArg,_arguments||[])).next())})};Object.defineProperty(exports,"__esModule",{value:!0}),exports.getEpisodes=void 0;const getEpisodes=function(_a){return __awaiter(this,arguments,void 0,function*({url:url,providerContext:providerContext}){const{axios:axios,cheerio:cheerio}=providerContext;try{const html=(yield axios.get(url)).data,$=cheerio.load(html),episodeLinks=[];return $("table tbody tr").each((i,element)=>{const linkElement=$(element).find("td:first-child a"),fileName=linkElement.text().trim(),fileLink=linkElement.attr("href");if(fileName&&fileLink&&"../"!==fileName&&"Parent Directory"!==fileName&&(fileName.includes(".mp4")||fileName.includes(".mkv")||fileName.includes(".avi")||fileName.includes(".mov"))){const fullLink=url+fileLink;let episodeTitle=fileName;const episodeMatch=fileName.match(/[Ss](\d+)[Ee](\d+)/),simpleEpisodeMatch=fileName.match(/[Ee](\d+)/);if(episodeMatch)episodeTitle=`S${episodeMatch[1]}E${episodeMatch[2]} - ${fileName}`;else if(simpleEpisodeMatch)episodeTitle=`Episode ${simpleEpisodeMatch[1]} - ${fileName}`;else{const numberMatch=fileName.match(/(\d+)/);numberMatch&&(episodeTitle=`Episode ${numberMatch[1]} - ${fileName}`)}episodeLinks.push({title:episodeTitle,link:fullLink})}}),episodeLinks}catch(err){return[]}})};exports.getEpisodes=getEpisodes; \ No newline at end of file diff --git a/dist/a111477/meta.js b/dist/a111477/meta.js new file mode 100644 index 0000000..a0ab14f --- /dev/null +++ b/dist/a111477/meta.js @@ -0,0 +1 @@ +"use strict";var __awaiter=this&&this.__awaiter||function(thisArg,_arguments,P,generator){return new(P||(P=Promise))(function(resolve,reject){function fulfilled(value){try{step(generator.next(value))}catch(e){reject(e)}}function rejected(value){try{step(generator.throw(value))}catch(e){reject(e)}}function step(result){var value;result.done?resolve(result.value):(value=result.value,value instanceof P?value:new P(function(resolve){resolve(value)})).then(fulfilled,rejected)}step((generator=generator.apply(thisArg,_arguments||[])).next())})};Object.defineProperty(exports,"__esModule",{value:!0}),exports.getMeta=void 0;const getMeta=function(_a){return __awaiter(this,arguments,void 0,function*({link:link,providerContext:providerContext}){try{const{axios:axios,cheerio:cheerio}=providerContext,url=link,data=(yield axios.get(url)).data,$=cheerio.load(data),title=($("h1").text().trim()||url.split("/").filter(Boolean).pop()||"").replace("Index of /","").replace(/\/$/,""),links=[],directLinks=[];$("table tbody tr").each((i,element)=>{const linkElement=$(element).find("td:first-child a"),itemTitle=linkElement.text().trim(),itemLink=linkElement.attr("href");if(itemTitle&&itemLink&&"../"!==itemTitle&&"Parent Directory"!==itemTitle){const fullLink=link+itemLink;if(itemTitle.endsWith("/")){const cleanTitle=itemTitle.replace(/\/$/,"");links.push({episodesLink:fullLink,title:cleanTitle})}else(itemTitle.includes(".mp4")||itemTitle.includes(".mkv")||itemTitle.includes(".avi")||itemTitle.includes(".mov"))&&directLinks.push({title:itemTitle,link:fullLink})}}),directLinks.length>0&&links.push({title:title+" (Direct Files)",directLinks:directLinks});const type=links.some(link=>{var _a,_b;return(null===(_a=link.episodesLink)||void 0===_a?void 0:_a.includes("Season"))||(null===(_b=link.episodesLink)||void 0===_b?void 0:_b.includes("S0"))})||directLinks.length>1?"series":"movie";return{title:title,synopsis:"Content from 111477.xyz directory",image:`https://placehold.jp/23/000000/ffffff/300x450.png?text=${encodeURIComponent(title)}&css=%7B%22background%22%3A%22%20-webkit-gradient(linear%2C%20left%20bottom%2C%20left%20top%2C%20from(%233f3b3b)%2C%20to(%23000000))%22%2C%22text-transform%22%3A%22%20capitalize%22%7D`,imdbId:"",type:type,linkList:links}}catch(err){return{title:"",synopsis:"",image:"",imdbId:"",type:"movie",linkList:[]}}})};exports.getMeta=getMeta; \ No newline at end of file diff --git a/dist/a111477/posts.js b/dist/a111477/posts.js new file mode 100644 index 0000000..34a5176 --- /dev/null +++ b/dist/a111477/posts.js @@ -0,0 +1 @@ +"use strict";var __awaiter=this&&this.__awaiter||function(thisArg,_arguments,P,generator){return new(P||(P=Promise))(function(resolve,reject){function fulfilled(value){try{step(generator.next(value))}catch(e){reject(e)}}function rejected(value){try{step(generator.throw(value))}catch(e){reject(e)}}function step(result){var value;result.done?resolve(result.value):(value=result.value,value instanceof P?value:new P(function(resolve){resolve(value)})).then(fulfilled,rejected)}step((generator=generator.apply(thisArg,_arguments||[])).next())})};Object.defineProperty(exports,"__esModule",{value:!0}),exports.getSearchPosts=exports.getPosts=void 0;const getPosts=function(_a){return __awaiter(this,arguments,void 0,function*({filter:filter,page:page,signal:signal,providerContext:providerContext}){const{axios:axios,cheerio:cheerio}=providerContext;if(page>1)return[];return posts({baseUrl:"https://a.111477.xyz",url:`https://a.111477.xyz${filter}`,signal:signal,axios:axios,cheerio:cheerio})})};exports.getPosts=getPosts;const getSearchPosts=function(_a){return __awaiter(this,arguments,void 0,function*({searchQuery:searchQuery,page:page,signal:signal,providerContext:providerContext}){const{axios:axios,cheerio:cheerio}=providerContext,baseUrl="https://a.111477.xyz";if(page>1)return[];return[...yield posts({baseUrl:baseUrl,url:`${baseUrl}/movies/`,signal:signal,axios:axios,cheerio:cheerio}),...yield posts({baseUrl:baseUrl,url:`${baseUrl}/tvs/`,signal:signal,axios:axios,cheerio:cheerio})].filter(post=>post.title.toLowerCase().includes(searchQuery.toLowerCase()))})};function posts(_a){return __awaiter(this,arguments,void 0,function*({baseUrl:baseUrl,url:url,signal:signal,axios:axios,cheerio:cheerio}){try{const data=(yield axios.get(url,{signal:signal})).data,$=cheerio.load(data),catalog=[];return $("table tbody tr").each((i,element)=>{const linkElement=$(element).find("td:first-child a"),title=linkElement.text().trim(),link=linkElement.attr("href");if(title&&link&&"../"!==title&&"Parent Directory"!==title&&title.endsWith("/")){const cleanTitle=title.replace(/\/$/,""),fullLink=url+link,imageTitle=cleanTitle.length>30?cleanTitle.slice(0,30).replace(/\./g," "):cleanTitle.replace(/\./g," "),image=`https://placehold.jp/23/000000/ffffff/200x400.png?text=${encodeURIComponent(imageTitle)}&css=%7B%22background%22%3A%22%20-webkit-gradient(linear%2C%20left%20bottom%2C%20left%20top%2C%20from(%233f3b3b)%2C%20to(%23000000))%22%2C%22text-transform%22%3A%22%20capitalize%22%7D`;catalog.push({title:cleanTitle,link:fullLink,image:image})}}),catalog}catch(err){return[]}})}exports.getSearchPosts=getSearchPosts; \ No newline at end of file diff --git a/dist/a111477/stream.js b/dist/a111477/stream.js new file mode 100644 index 0000000..1aacbbf --- /dev/null +++ b/dist/a111477/stream.js @@ -0,0 +1 @@ +"use strict";var __awaiter=this&&this.__awaiter||function(thisArg,_arguments,P,generator){return new(P||(P=Promise))(function(resolve,reject){function fulfilled(value){try{step(generator.next(value))}catch(e){reject(e)}}function rejected(value){try{step(generator.throw(value))}catch(e){reject(e)}}function step(result){var value;result.done?resolve(result.value):(value=result.value,value instanceof P?value:new P(function(resolve){resolve(value)})).then(fulfilled,rejected)}step((generator=generator.apply(thisArg,_arguments||[])).next())})};Object.defineProperty(exports,"__esModule",{value:!0}),exports.getStream=void 0;const getStream=function(_a){return __awaiter(this,arguments,void 0,function*({link:url}){var _b;try{const stream=[],fileExtension=(null===(_b=url.split(".").pop())||void 0===_b?void 0:_b.toLowerCase())||"mp4";let streamType="mp4";return["mkv","avi","mov","webm"].includes(fileExtension)&&(streamType=fileExtension),stream.push({server:"111477.xyz",link:url,type:streamType,headers:{"User-Agent":"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.124 Safari/537.36",Referer:"https://a.111477.xyz/"}}),stream}catch(err){return[]}})};exports.getStream=getStream; \ No newline at end of file diff --git a/manifest.json b/manifest.json index 1de49b4..109fceb 100644 --- a/manifest.json +++ b/manifest.json @@ -135,13 +135,21 @@ "type": "global", "disabled": false }, + { + "display_name": "a.111477", + "value": "a111477", + "version": "1.0", + "icon": "", + "type": "english", + "disabled": false + }, { "display_name": "VadaPav", "value": "vadapav", "version": "1.0", "icon": "", "type": "global", - "disabled": false + "disabled": true }, { "display_name": "MoviesApi", diff --git a/providers/a111477/catalog.ts b/providers/a111477/catalog.ts new file mode 100644 index 0000000..da12ab5 --- /dev/null +++ b/providers/a111477/catalog.ts @@ -0,0 +1,24 @@ +export const catalog = [ + { + title: "Movies", + filter: "/movies/", + }, + { + title: "TV Shows", + filter: "/tvs/", + }, + { + title: "K-Drama", + filter: "/kdrama/", + }, + { + title: "Asian Drama", + filter: "/asiandrama/", + }, + { + title: "Misc", + filter: "/misc/", + }, +]; + +export const genres = []; diff --git a/providers/a111477/episodes.ts b/providers/a111477/episodes.ts new file mode 100644 index 0000000..ccc58f3 --- /dev/null +++ b/providers/a111477/episodes.ts @@ -0,0 +1,69 @@ +import { EpisodeLink, ProviderContext } from "../types"; + +export const getEpisodes = async function ({ + url, + providerContext, +}: { + url: string; + providerContext: ProviderContext; +}): Promise { + const { axios, cheerio } = providerContext; + try { + const res = await axios.get(url); + const html = res.data; + const $ = cheerio.load(html); + const episodeLinks: EpisodeLink[] = []; + + // Parse episode files from directory + $("table tbody tr").each((i, element) => { + const $row = $(element); + const linkElement = $row.find("td:first-child a"); + const fileName = linkElement.text().trim(); + const fileLink = linkElement.attr("href"); + + if ( + fileName && + fileLink && + fileName !== "../" && + fileName !== "Parent Directory" + ) { + // Check if it's a video file + if ( + fileName.includes(".mp4") || + fileName.includes(".mkv") || + fileName.includes(".avi") || + fileName.includes(".mov") + ) { + const fullLink = url + fileLink; + + // Try to extract episode information from filename + let episodeTitle = fileName; + const episodeMatch = fileName.match(/[Ss](\d+)[Ee](\d+)/); + const simpleEpisodeMatch = fileName.match(/[Ee](\d+)/); + + if (episodeMatch) { + episodeTitle = `S${episodeMatch[1]}E${episodeMatch[2]} - ${fileName}`; + } else if (simpleEpisodeMatch) { + episodeTitle = `Episode ${simpleEpisodeMatch[1]} - ${fileName}`; + } else { + // Try to extract episode number from various patterns + const numberMatch = fileName.match(/(\d+)/); + if (numberMatch) { + episodeTitle = `Episode ${numberMatch[1]} - ${fileName}`; + } + } + + episodeLinks.push({ + title: episodeTitle, + link: fullLink, + }); + } + } + }); + + return episodeLinks; + } catch (err) { + console.error("111477 episodes error:", err); + return []; + } +}; diff --git a/providers/a111477/meta.ts b/providers/a111477/meta.ts new file mode 100644 index 0000000..4a2ae50 --- /dev/null +++ b/providers/a111477/meta.ts @@ -0,0 +1,103 @@ +import { EpisodeLink, Info, Link, ProviderContext } from "../types"; + +export const getMeta = async function ({ + link, + providerContext, +}: { + link: string; + providerContext: ProviderContext; +}): Promise { + try { + const { axios, cheerio } = providerContext; + const url = link; + const res = await axios.get(url); + const data = res.data; + const $ = cheerio.load(data); + + // Extract title from the page header or URL + const pageTitle = + $("h1").text().trim() || url.split("/").filter(Boolean).pop() || ""; + const title = pageTitle.replace("Index of /", "").replace(/\/$/, ""); + + const links: Link[] = []; + const directLinks: EpisodeLink[] = []; + + // Parse directory structure + $("table tbody tr").each((i, element) => { + const $row = $(element); + const linkElement = $row.find("td:first-child a"); + const itemTitle = linkElement.text().trim(); + const itemLink = linkElement.attr("href"); + + if ( + itemTitle && + itemLink && + itemTitle !== "../" && + itemTitle !== "Parent Directory" + ) { + const fullLink = link + itemLink; + + // If it's a directory (ends with /) + if (itemTitle.endsWith("/")) { + const cleanTitle = itemTitle.replace(/\/$/, ""); + links.push({ + episodesLink: fullLink, + title: cleanTitle, + }); + } + // If it's a video file + else if ( + itemTitle.includes(".mp4") || + itemTitle.includes(".mkv") || + itemTitle.includes(".avi") || + itemTitle.includes(".mov") + ) { + directLinks.push({ + title: itemTitle, + link: fullLink, + }); + } + } + }); + + // If there are direct video files, add them as a direct link group + if (directLinks.length > 0) { + links.push({ + title: title + " (Direct Files)", + directLinks: directLinks, + }); + } + + // Determine if this is a movie or series based on structure + const type = links.some( + (link) => + link.episodesLink?.includes("Season") || + link.episodesLink?.includes("S0") + ) + ? "series" + : directLinks.length > 1 + ? "series" + : "movie"; + + return { + title: title, + synopsis: `Content from 111477.xyz directory`, + image: `https://placehold.jp/23/000000/ffffff/300x450.png?text=${encodeURIComponent( + title + )}&css=%7B%22background%22%3A%22%20-webkit-gradient(linear%2C%20left%20bottom%2C%20left%20top%2C%20from(%233f3b3b)%2C%20to(%23000000))%22%2C%22text-transform%22%3A%22%20capitalize%22%7D`, + imdbId: "", + type: type, + linkList: links, + }; + } catch (err) { + console.error("111477 meta error:", err); + return { + title: "", + synopsis: "", + image: "", + imdbId: "", + type: "movie", + linkList: [], + }; + } +}; diff --git a/providers/a111477/posts.ts b/providers/a111477/posts.ts new file mode 100644 index 0000000..ff0357f --- /dev/null +++ b/providers/a111477/posts.ts @@ -0,0 +1,128 @@ +import { Post, ProviderContext } from "../types"; + +export const getPosts = async function ({ + filter, + page, + signal, + providerContext, +}: { + filter: string; + page: number; + providerValue: string; + signal: AbortSignal; + providerContext: ProviderContext; +}): Promise { + const { axios, cheerio } = providerContext; + const baseUrl = "https://a.111477.xyz"; + if (page > 1) { + return []; + } + const url = `${baseUrl}${filter}`; + return posts({ baseUrl, url, signal, axios, cheerio }); +}; + +export const getSearchPosts = async function ({ + searchQuery, + page, + signal, + providerContext, +}: { + searchQuery: string; + page: number; + providerValue: string; + signal: AbortSignal; + providerContext: ProviderContext; +}): Promise { + const { axios, cheerio } = providerContext; + const baseUrl = "https://a.111477.xyz"; + if (page > 1) { + return []; + } + + // Search through both movies and TV shows directories + const moviesPosts = await posts({ + baseUrl, + url: `${baseUrl}/movies/`, + signal, + axios, + cheerio, + }); + const tvsPosts = await posts({ + baseUrl, + url: `${baseUrl}/tvs/`, + signal, + axios, + cheerio, + }); + + // Combine all posts + const allPosts = [...moviesPosts, ...tvsPosts]; + + // Filter posts based on search query + const filteredPosts = allPosts.filter((post) => + post.title.toLowerCase().includes(searchQuery.toLowerCase()) + ); + + return filteredPosts; +}; + +async function posts({ + baseUrl, + url, + signal, + axios, + cheerio, +}: { + baseUrl: string; + url: string; + signal: AbortSignal; + axios: ProviderContext["axios"]; + cheerio: ProviderContext["cheerio"]; +}): Promise { + try { + const res = await axios.get(url, { signal }); + const data = res.data; + const $ = cheerio.load(data); + const catalog: Post[] = []; + + // Parse the directory listing + $("table tbody tr").each((i, element) => { + const $row = $(element); + const linkElement = $row.find("td:first-child a"); + const title = linkElement.text().trim(); + const link = linkElement.attr("href"); + + // Skip parent directory and files, only get folders + if ( + title && + link && + title !== "../" && + title !== "Parent Directory" && + title.endsWith("/") + ) { + const cleanTitle = title.replace(/\/$/, ""); // Remove trailing slash + const fullLink = url + link; + + // Generate a placeholder image based on title + const imageTitle = + cleanTitle.length > 30 + ? cleanTitle.slice(0, 30).replace(/\./g, " ") + : cleanTitle.replace(/\./g, " "); + const image = `https://placehold.jp/23/000000/ffffff/200x400.png?text=${encodeURIComponent( + imageTitle + )}&css=%7B%22background%22%3A%22%20-webkit-gradient(linear%2C%20left%20bottom%2C%20left%20top%2C%20from(%233f3b3b)%2C%20to(%23000000))%22%2C%22text-transform%22%3A%22%20capitalize%22%7D`; + + catalog.push({ + title: cleanTitle, + link: fullLink, + image: image, + }); + } + }); + + return catalog; + } catch (err) { + console.error("111477 directory listing error:", err); + return []; + } +} diff --git a/providers/a111477/stream.ts b/providers/a111477/stream.ts new file mode 100644 index 0000000..efd290c --- /dev/null +++ b/providers/a111477/stream.ts @@ -0,0 +1,38 @@ +import { Stream, ProviderContext } from "../types"; + +export const getStream = async function ({ + link: url, +}: { + link: string; + type: string; + providerContext: ProviderContext; +}): Promise { + try { + const stream: Stream[] = []; + + // Get file extension from URL + const fileExtension = url.split(".").pop()?.toLowerCase() || "mp4"; + + // Determine stream type based on file extension + let streamType = "mp4"; + if (["mkv", "avi", "mov", "webm"].includes(fileExtension)) { + streamType = fileExtension; + } + + stream.push({ + server: "111477.xyz", + link: url, + type: streamType, + headers: { + "User-Agent": + "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.124 Safari/537.36", + Referer: "https://a.111477.xyz/", + }, + }); + + return stream; + } catch (err) { + console.error("111477 stream error:", err); + return []; + } +};