Greasy Fork is available in English.
Toggle for Searching Torrents via Search aggegrator
当前为
// ==UserScript==
// @name Torrent Quick Search
// @namespace https://github.com/TMD20/torrent-quick-search
// @supportURL https://github.com/TMD20/torrent-quick-search
// @version 1.40
// @description Toggle for Searching Torrents via Search aggegrator
// @icon https://cdn2.iconfinder.com/data/icons/flat-icons-19/512/Eye.png
// @author tmd
// @noframes
// @inject-into page
// @run-at document-end
// @require https://openuserjs.org/src/libs/sizzle/GM_config.min.js
// @grant GM_getValue
// @grant GM_setValue
// @grant GM.xmlhttpRequest
// @grant GM.registerMenuCommand
// @grant GM.notification
// @match https://animebytes.tv/requests.php?action=viewrequest&id=*
// @match https://animebytes.tv/series.php?id=*
// @match https://animebytes.tv/torrents.php?id=*
// @match https://blutopia.xyz/requests/*
// @match https://blutopia.xyz/torrents/*
// @match https://beyond-hd.me/requests/*
// @match https://beyond-hd.me/torrents/*
// @match https://beyond-hd.me/library/title/*
// @match https://imdb.com/title/*
//
// @match https://www.imdb.com/title/*
// @match https://www.themoviedb.org/movie/*
// @match https://www.themoviedb.org/tv/*
// @license MIT
// ==/UserScript==
`
General Functions
Functions that don't fit in any other catergory
`
function recreateController()
{
controller = new AbortController();
}
searchObj = {
ready: true,
search()
{
if (controller.signal.aborted)
{
return Promise.reject(AbortError)
}
return new Promise(async (resolve, reject) =>
{
controller.signal.addEventListener("abort", () =>
{
reject(AbortError)
});
document.querySelector("#torrent-quicksearch-msgnode").textContent = "Loading"
indexers = await getIndexers()
imdb=await setIMDBNode()
setTitleNode()
//reset count
let count = []
let length = indexers.length
let data = []
x = 5
while (indexers.length)
{
// x at a time
newData = await Promise.allSettled(indexers.splice(0, Math.min(indexers.length, x)).map((e) => searchIndexer(e, imdb, length, count)))
data = [...data, ...newData]
}
console.log(data)
errorMsgs = data.filter((e) => e["status"] == "rejected").map((e) => e["reason"].message)
errorMsgs = [...new Set(errorMsgs)]
if (errorMsgs.length > 0)
{
reject(errorMsgs.join("\n"))
}
addNumbers()
resolve()
}
)
},
cancel()
{
controller.abort()
},
async setup()
{
this.searchPromise = new Promise((resolve, reject) =>
{
this.timeout = setTimeout(async () =>
{
try
{
resolve(await this.search())
}
catch (e)
{
reject(e)
}
}, 1000)
})
},
async doSearch()
{
showDisplay()
recreateController()
await this.setup()
setTimeout(async()=>{
resetResultList()
resetSearchDOM()
getTableHead()
try{
await this.searchPromise
}
catch (error)
{
if (error.message.match(/aborted!/i) === null)
{
GM.notification(error.message, program, searchIcon)
}
console.log(error)
}
},0)
},
finalize()
{
if (Array.from(document.querySelectorAll(".torrent-quicksearch-resultitem")).length == 0)
{
this.nomatchID = setTimeout(() => document.querySelector("#torrent-quicksearch-resultlist").textContent = "No Matches", 1000)
}
this.finalmsgID = setTimeout(() => document.querySelector("#torrent-quicksearch-msgnode").textContent = "Finished", 1000)
this.removemsgnodeID = setTimeout(() => document.querySelector("#torrent-quicksearch-msgnode").style.display = "none", 3000)
},
async toggleSearch()
{
content = document.querySelector("#torrent-quicksearch-box")
if (content.style.display === "inline-block")
{
hideDisplay()
searchObj.cancel()
}
else if ((content.style.display === "none" || content.style.display === ""))
{
customSearch = false
await this.doSearch()
}
}
}
function searchIndexer(indexerObj, imdb, total, count)
{
if (controller.signal.aborted)
{
return Promise.reject(AbortError)
}
return new Promise(async (resolve, reject) =>
{
msg = null
controller.signal.addEventListener("abort", () =>
{
reject(AbortError)
});
searchprograms = GM_config.get('searchprogram')
let data = null
if (searchprogram == "Prowlarr")
{
data = await searchProwlarrIndexer(indexerObj, controller)
}
else if (searchprogram == "Jackett")
{
data = await searchJackettIndexer(indexerObj)
}
else if (searchprogram == "NZBHydra2")
{
data = await searchHydra2Indexer(indexerObj)
}
msg = `Results fetched fom ${indexerObj["name"]}:${count.length+1}/${total} Indexers completed`
data = data.filter((e) => imdbFilter(e, imdbCleanup(imdb)))
data.forEach((e) =>
{
if (e["ImdbId"] == 0 || e["ImdbId"] == null)
{
e["ImdbId"] = "Not Provided"
}
})
data = data.filter((e) => currSiteFilter(e["infoUrl"]))
addResultsTable(data)
count.push(indexerObj["id"])
document.querySelector("#torrent-quicksearch-msgnode").textContent = msg
console.log(msg)
resolve(data)
})
}
async function searchProwlarrIndexer(indexer)
{
req = await fetch(getSearchURLProwlarr(indexer["id"]),
{
"timeout": indexerSearchTimeout
})
data = JSON.parse(req.responseText)
promiseArray = await Promise.allSettled(data.map(async (e) =>
{
return {
"Title": e["title"],
"Indexer": e["indexer"],
"Grabs": e["grabs"],
"PublishDate": e["publishDate"],
"Size": e["size"],
"Leechers": e["leechers"],
"Seeders": e["seeders"],
"InfoUrl": e["details"],
"DownloadUrl": e["downloadUrl"],
"ImdbId": e["imdbId"],
"TvdbId": await tmdbTVDBConvertor(e["imdbId"]),
"Cost": e["indexerFlags"].includes("freeleech") == "100% Freeleech" ? "100% Freeleech" : "Cost Unknown With Prowlarr",
"Protocol": e["protocol"],
}
}))
return promiseArray.map((e) => e["value"]).filter((e) => e != null)
}
async function searchJackettIndexer(indexer)
{
req = await fetch(getSearchURLJackett(indexer["id"]),
{
"timeout": indexerSearchTimeout
})
data = JSON.parse(req.responseText)["Results"]
promiseArray = await Promise.allSettled(data.map(async (e) =>
{
return {
"Title": e["Title"],
"Indexer": e["Tracker"],
"Grabs": e["Grabs"],
"PublishDate": e["PublishDate"],
"Size": e["Size"],
"Leechers": e["Peers"],
"Seeders": e["Seeders"],
"InfoUrl": e["Details"],
"DownloadUrl": e["Link"],
"ImdbId": e["Imdb"],
"TvdbId": await tmdbTVDBConvertor(e["Imdb"]),
"Cost": `${(1-e["DownloadVolumeFactor"])*100}% Freeleech`,
"Protocol": "torrent",
}
}))
return promiseArray.map((e) => e["value"]).filter((e) => e != null)
}
async function searchHydra2Indexer(indexer)
{
req = await fetch(getSearchURLHydraTor(indexer["id"]),
{
"timeout": indexerSearchTimeout
})
req2 = await fetch(getSearchURLHydraNZB(indexer["id"]),
{
"timeout": indexerSearchTimeout
})
parser = new DOMParser();
data = [...Array.from(parser.parseFromString(req.responseText, "text/xml").querySelectorAll("channel>item")), ...Array.from(parser.parseFromString(req2.responseText, "text/xml").querySelectorAll("channel>item"))]
//array is final dictkey,queryselector,attribute
promiseArray = await Promise.allSettled(
data.map(async (e) =>
{
t = [
["Title", "title", "textContent"],
["Indexer", "[name=hydraIndexerName]", "null"],
["Leechers", "[name=peers]", "null"],
["Seeders", "[name=seeders]", "null"],
["Cost", "[name=downloadvolumefactor]", "null"],
["PublishDate", "pubDate", "textContent"],
["Size", "size", "textContent"],
["InfoUrl", "comments", "textContent"],
["DownloadUrl", "link", "textContent"],
["ImdbId", "[name=imdb]", "null"]
]
out = {}
out["Grabs"] = "Hydra Does not Report"
for (i in t)
{
key = t[i][0]
node = e.querySelector(t[i][1])
textContent = (t[i][2] == "textContent")
if (!node)
{
continue
}
if (textContent)
{
out[key] = node.textContent
}
else if (key == "cost")
{
out[key] = `${(1-node.getAttribute("value"))*100}% Freeleech`
}
else
{
out[key] = node.getAttribute("value")
}
}
out["TvdbId"] = await tmdbTVDBConvertor(out["ImdbId"])
out["Protocol"] = data[0].querySelector("enclosure").getAttribute("type") == "application/x-bittorrent" ? "torrent" : "usenet"
return out
})
)
return promiseArray.map((e) => e["value"]).filter((e) => e != null)
}
function fetch(url,
{
method = "GET",
data = null,
headers = {},
timeout = 300000
} = {})
{
return new Promise((resolve, reject) =>
{
controller.signal.addEventListener("abort", () =>
{
reject(AbortError)
});
setTimeout(() => reject(AbortError), timeout)
GM.xmlhttpRequest(
{
'method': method,
'url': url,
'data': data,
'headers': headers,
onload: response =>
{
resolve(response)
},
onerror: response =>
{
reject(response.responseText)
},
})
}
)
return req
}
function getParser()
{
siteName = standardNames[window.location.host] || window.location.host
data = infoParser[siteName]
if (data === undefined)
{
msg = "Could not get Parser"
GM.notification(msg, program, searchIcon)
throw new Error(msg);
}
return data
}
function verifyConfig()
{
if (GM_config.get('searchapi', "null") == "null" || GM_config.get('searchurl', "null") == "null")
{
return false
}
if (GM_config.get('searchapi', "null") == "" || GM_config.get('searchurl', "null") == "")
{
return false
}
return true
}
`
DOM Manipulators
These Functions are used to manipulate the DOM
`
function setTitleNode(){
if (customSearch == false)
{
document.querySelector("#torrent-quicksearch-customsearch").value = getTitle()
}
}
async function setIMDBNode(){
imdb = null
document.querySelector("#torrent-quicksearch-msgnode").textContent = "Fetching Results From Indexers"
//Get Old IMDB
if (document.querySelector("#torrent-quicksearch-imdbinfo").textContent != imdbParserFail &&
document.querySelector("#torrent-quicksearch-imdbinfo").textContent.length != 0 &&
document.querySelector("#torrent-quicksearch-imdbinfo").textContent != "None"
)
{
imdb = document.querySelector("#torrent-quicksearch-imdbinfo").textContent
}
//Else get New IMDB
else
{
imdb = await getIMDB()
document.querySelector("#torrent-quicksearch-imdbinfo").textContent = imdb || imdbParserFail
}
return imdb
}
function resetSearchDOM()
{
document.querySelector("#torrent-quicksearch-imdbinfo").textContent = "None"
document.querySelector("#torrent-quicksearch-msgnode").textContent = ""
}
function hideDisplay()
{
document.querySelector("#torrent-quicksearch-overlay").style.setProperty('--icon-size', `${iconSmall}%`);
document.querySelector("#torrent-quicksearch-customsearch").value = ""
content.style.display = "none"
}
function showDisplay()
{
document.querySelector("#torrent-quicksearch-msgnode").textContent = ""
document.querySelector("#torrent-quicksearch-msgnode").style.display = "block"
document.querySelector("#torrent-quicksearch-overlay").style.setProperty('--icon-size', `${iconLarge}%`);
content.style.display = "inline-block";
}
function getTableHead()
{
node = document.querySelector("#torrent-quicksearch-resultheader");
node.innerHTML = `
<span class="torrent-quicksearch-resultcell" >Links</span>
<span class="torrent-quicksearch-resultcell" >Arr</span>
<span class="torrent-quicksearch-resultcell">Number</span>
<span class="torrent-quicksearch-resultcell" >Title</span>
<span class="torrent-quicksearch-resultcell" >Indexer</span>
<span class="torrent-quicksearch-resultcell" >Grabs</span>
<span class="torrent-quicksearch-resultcell" >Seeders</span>
<span class="torrent-quicksearch-resultcell" >Leechers</span>
<span class="torrent-quicksearch-resultcell" >DLCost</span>
<span class="torrent-quicksearch-resultcell" >Date</span>
<span class="torrent-quicksearch-resultcell">Size</span>
<span class="torrent-quicksearch-resultcell">IMDB</span>
`
Array.from(node.children).forEach((e, i) =>
{
e.style.gridColumnStart = i + 1
e.style.fontSize = `${GM_config.get("fontsize",12)}px`
})
}
function addResultsTable(data)
{
if (data.length == 0)
{
return
}
resultList = document.querySelector("#torrent-quicksearch-resultlist")
tempFrag = new DocumentFragment()
data.forEach((e, i) =>
{
node = document.createElement("span");
node.setAttribute("class", "torrent-quicksearch-resultitem")
node.innerHTML = `
<span class="torrent-quicksearch-resultcell" style='font-size:${GM_config.get("fontsize",12)};grid-column-start:2' >
<a href=${e['DownloadUrl']}>Download</a>
<br>
<br>
<a href=${e['InfoUrl']}>Details</a>
</span>
<span class="torrent-quicksearch-resultcell" style='font-size:${GM_config.get("fontsize",12)};grid-column-start:3'>
<a>Send to Sonarr</a>
<br>
<br>
<a>Send to Radarr</a>
</span>
<span class="torrent-quicksearch-resultcell" style='font-size:${GM_config.get("fontsize",12)};grid-column-start:4' >?</span>
<span class="torrent-quicksearch-resultcell" style='font-size:${GM_config.get("fontsize",12)};grid-column-start:5' >${e['Title']}</span>
<span class="torrent-quicksearch-resultcell" style='font-size:${GM_config.get("fontsize",12)};grid-column-start:6' >${e['Indexer']}</span>
<span class="torrent-quicksearch-resultcell" style='font-size:${GM_config.get("fontsize",12)};grid-column-start:7'>${e['Grabs']||"No Data"} </span>
<span class="torrent-quicksearch-resultcell" style='font-size:${GM_config.get("fontsize",12)};grid-column-start:8'>${e['Seeders']||"No Data"} </span>
<span class="torrent-quicksearch-resultcell" style='font-size:${GM_config.get("fontsize",12)};grid-column-start:9' >${e['Leechers']||"No Data"} </span>
<span class="torrent-quicksearch-resultcell" style='font-size:${GM_config.get("fontsize",12)};grid-column-start:10'>${e['Cost']} </span>
<span class="torrent-quicksearch-resultcell" style='font-size:${GM_config.get("fontsize",12)};grid-column-start:11' >${new Date(e['PublishDate']).toLocaleString("en-CA")}</span>
<span class="torrent-quicksearch-resultcell" style='font-size:${GM_config.get("fontsize",12)};grid-column-start:12' >${(parseInt(e['Size'])/1073741824).toFixed(2)} GB</span>
<span class="torrent-quicksearch-resultcell" style='font-size:${GM_config.get("fontsize",12)};grid-column-start:13' >${e['ImdbId']}</span>`
processSonarrNode(node, e)
processRadarrNode(node, e)
tempFrag.append(node)
})
resultList.appendChild(tempFrag)
}
function resetResultList()
{
document.querySelector("#torrent-quicksearch-resultheader").textContent = ""
document.querySelector("#torrent-quicksearch-resultlist").textContent = ""
}
function addNumbers()
{
Array.from(document.querySelectorAll(".torrent-quicksearch-resultitem")).forEach((e, i) =>
{
node = Array.from(e.children).filter((e) => e["textContent"] == "?")[0]
node.textContent = `${i+1}`
}
)
}
async function processSonarrNode(node, data)
{
node = Array.from(node.querySelectorAll("span>span>a")).filter((e) => e["textContent"] == "Send to Sonarr")[0]
if (GM_config.get('sonarrurl') === "null" ||
GM_config.get('sonarrurl') === "" ||
GM_config.get('sonarrapi') === "null" ||
GM_config.get('sonarrapi') === "")
{
node.remove()
return
}
nodeEvent = await sonarrFactory(data)
node.addEventListener("click", nodeEvent)
node.style.cursor = "pointer"
}
async function processRadarrNode(node, data)
{
node = Array.from(node.querySelectorAll("span>span>a")).filter((e) => e["textContent"] == "Send to Radarr")[0]
if (GM_config.get('radarrurl') === "null" ||
GM_config.get('radarrurl') === "" ||
GM_config.get('radarrapi') === "null" ||
GM_config.get('radarrapi') === ""
)
{
node.remove()
return
}
nodeEvent = await radarrFactory(data)
node.addEventListener("click", nodeEvent)
node.style.cursor = "pointer"
}
function createMainDOM()
{
const box = document.createElement("div");
box.setAttribute("id", "torrent-quicksearch-overlay");
rowSplit = 12
contentWidth = 70
boxMinHeight = 5
boxMaxHeight = 100
boxHeight = 40
boxWidth = 70
boxMaxWidth = 150
box.innerHTML = `
<div>
<img id="torrent-quicksearch-toggle" src="${searchIcon}"></img>
<div id="torrent-quicksearch-box">
<div id="torrent-quicksearch-content">
<div id="torrent-quicksearch-msgnode"></div>
<div id="torrent-quicksearch-custombox">
<div>
<label>Title:</label>
<input type="text" id="torrent-quicksearch-customsearch">
<label>IMDB:</label>
<div id="torrent-quicksearch-imdbinfo">None</div>
<button id="torrent-quicksearch-customsearchbutton">Custom Search</button>
</div>
</div>
<div id="torrent-quicksearch-resultheader"></div>
<div id="torrent-quicksearch-resultlist">
</div>
</div>
</div>
<style>
/* Variables */
#torrent-quicksearch-overlay {
--grid-size: max(calc(50vw/${rowSplit}),calc(100%/${rowSplit}));
--icon-size:${iconSmall}%;
--icon-padding:${paddingSmall}%;
}
#torrent-quicksearch-overlay {
position: sticky;
display: flex;
flex-direction: column;
gap: 10px;
top: 40vh;
pointer-events: none;
z-index: 900000;
}
#torrent-quicksearch-overlay> div:first-of-type {
position: absolute;
left:80vw;
}
#torrent-quicksearch-toggle {
margin-left: auto;
display:block;
cursor: pointer;
pointer-events:all;
width: var(--icon-size);
height: var(--icon-size);
padding-top: var(--icon-padding);
padding-bottom:var(--icon-padding);
margin-bottom:calc(${paddingLarge}vh - var(--icon-padding));
margin-top:calc(${paddingLarge}vh - var(--icon-padding));
}
#torrent-quicksearch-box{
resize:both;
direction:rtl;
right:5vw;
margin-right:auto;
position:absolute;
display:none;
min-height: ${boxMinHeight}vh;
max-height:${boxMaxHeight}vh;
height: ${boxHeight}vh;
width: ${boxWidth}vw;
max-width: ${boxMaxWidth}vw;
overflow:hidden;
border:solid black 5px;
}
#torrent-quicksearch-msgnode{
background-color:#FFFFFF;
width:calc(var(--grid-size)*${rowSplit});
display:none;
}
#torrent-quicksearch-custombox {
background-color:#FFFFFF;
width:calc(var(--grid-size)*${rowSplit});
pointer-events:all;
}
#torrent-quicksearch-custombox>div {
display: flex;
background-color:#FFFFFF;
flex-direction:row;
justify-content: center;
width: 100%
}
#torrent-quicksearch-custombox>div >label {
margin-left:2.5%;
margin-right:2.5%;
}
#torrent-quicksearch-custombox>div >button {
margin-left:2.5%;
}
#torrent-quicksearch-customsearch{
background-color:#FFFFFF;
border:solid black 2px;
flex-grow:1;
}
#torrent-quicksearch-custombox > label{
margin-left:2%
margin-right:2%
}
#torrent-quicksearch-customsearchbutton {
background-color: #4CAF50; /* Green */
border: none;
color: white;
text-align: center;
text-decoration: none;
font-size: 16px;
}
#torrent-quicksearch-content {
pointer-events:all;
background-color: #d5cbcb;
scrollbar-color: white;
direction:ltr;
overflow:scroll;
height:100%;
width:100%;
}
#torrent-quicksearch-resultlist{
border:solid white 5px;
width:calc(var(--grid-size)*${rowSplit});
}
.torrent-quicksearch-resultitem,#torrent-quicksearch-resultheader{
display: grid;
grid-template-columns: repeat(${rowSplit},var(--grid-size));
width:calc((var(--grid-size)*${rowSplit})-10);
}
#torrent-quicksearch-resultlist>.torrent-quicksearch-resultitem:nth-child(even) {
background-color: #D7C49EFF;
}
#torrent-quicksearch-resultlist>.torrent-quicksearch-resultitem:nth-child(odd) {
background-color: #d5cbcb;
}
#torrent-quicksearch-resultheader{
background-color: #B1D79E;
}
.torrent-quicksearch-resultcell{
font-weight: bold;
margin-left: 10%;
overflow-wrap:break-word;
}
::-webkit-scrollbar-thumb{
background-color:white;
}
<style/>`
box.querySelector("#torrent-quicksearch-toggle").addEventListener("mousedown", leftClickProcess)
box.querySelector("#torrent-quicksearch-toggle").addEventListener("mouseup", mouseUpProcess)
document.addEventListener("mouseup", resetMouse)
box.querySelector("#torrent-quicksearch-customsearchbutton").addEventListener("click", () =>
{
searchObj.cancel()
setTimeout(()=>{
if (Date.now() - lastClick < clickLimit)
{
return
}
lastClick=Date.now()
customSearch = true
searchObj.doSearch()
},0)
})
document.body.insertBefore(box, document.body.children[0])
}
`
Matching Function
These help with finding a Match
`
function getTitle()
{
titleNode = document.querySelector(siteParser["title"])
if (titleNode == null)
{
throw new Error("Title Node Not Found")
}
title = titleNode[siteParser["titleAttrib"]]
title = titleCleanup(title)
return title
}
async function getIMDB()
{
let imdb = null
if (standardNames[window.location.host] == "imdb.com")
{
imdb = window.location.href
}
else if (standardNames[window.location.host] == "themoviedb.org")
{
imdb = tmdbPageIMDBParser()
}
else
{
imdbNode = document.querySelector(siteParser["imdb"])
if (imdbNode == null)
{
return null
}
imdb = imdbNode[siteParser["imdbAttrib"]]
}
imdb = imdbCleanup(imdb)
return imdb
}
function titleCleanup(title)
{
title = title.trim().replaceAll(/\n/g, "")
title = title.replaceAll(/ +/g, " ")
return title
}
function imdbCleanup(imdb)
{
if (imdb === null ||
imdb === undefined ||
imdb === 0 ||
imdb === "0")
{
return imdb
}
imdb = String(imdb)
imdb = imdb.match(/[0-9]+/).toString()
imdb = imdb.trim().replaceAll(/\n/g, "")
imdb = imdb.replace(/imdb/i, "")
imdb = imdb.replace(/tt/, "")
imdb = imdb.replace(/[:|&|*|\(|\)|!|@|#|$|%|^|\*|\\|\/]/, "")
imdb = imdb.replaceAll(/ +/g, "")
imdb = imdb.replace(/^0+/, "")
imdb = parseInt(imdb)
return imdb
}
function imdbFilter(entry, imdb)
{
if (imdb === null || imdb === "IMDB Not Provided")
{
return true
}
else if (entry["ImdbId"] == 0)
{
return true
}
else if (entry["ImdbId"] == imdb)
{
return true
}
return false
}
async function tmdbExternalMedia(type, id)
{
key = GM_config.get('tmdbapi', "null")
if (key == "null")
{
return null
}
let baseURL = `https://api.themoviedb.org/3/movie/${id}/external_ids`
let params = new URLSearchParams();
params.append("api_key", key)
if (type == "tv")
{
baseURL = `https://api.themoviedb.org/3/tv/${id}/external_ids`
}
searchURL = `${baseURL}?${params.toString()}`
req = await fetch(searchURL)
return JSON.parse(req.responseText)
}
async function imdbTMDBConvertor(imdb)
{
key = GM_config.get('tmdbapi', "null")
imdb = String(imdb)
if (key == "null")
{
return null
}
if (imdb.match(/tt/i) == null)
{
imdb = `tt${imdb}`
}
let baseURL = `https://api.themoviedb.org/3/find/${imdb}`
let params = new URLSearchParams();
params.append("api_key", key)
params.append("external_source", "imdb_id")
searchURL = `${baseURL}?${params.toString()}`
req = await fetch(searchURL)
return JSON.parse(req.responseText)
}
// First call to tmdbapi should be removed once we parse Movies vs TV
async function tmdbTVDBConvertor(imdb)
{
key = GM_config.get('tmdbapi', "null")
if (key == "null")
{
return null
}
helperData = await imdbTMDBConvertor(imdb)
let data = null
if (helperData["tv_results"].length > 0)
{
return (await tmdbExternalMedia("tv", helperData["tv_results"][0]["id"]))["tvdb_id"]
}
else if (helperData["movie_results"].length > 0)
{
return (await tmdbExternalMedia("movie", helperData["movie_results"][0]["id"]))["tvdb_id"]
}
}
async function tmdbPageIMDBParser()
{
id = window.location.href.match(/\/[0-9]+/).toString().substring(1)
if (window.location.href.match(/\/tv\//))
{
return await tmdbExternalMedia("tv", id)
}
else
{
return await tmdbExternalMedia("movie", id)["imdb_id"]
}
}
function currSiteFilter(entryURL)
{
if (GM_config.get('sitefilter') == "false")
{
return true
}
if (new URL(entryURL).hostname.replace(/\..*$/, "") == window.location.host.replace(/\..*$/, ""))
{
return false
}
return true
}
`
URL Processing Indexer
Functions Used to Get Indexer Info
`
async function getIndexers()
{
document.querySelector("#torrent-quicksearch-msgnode").textContent = "Getting Indexers"
searchprogram = GM_config.get('searchprogram')
indexers = null
if (searchprogram == "Prowlarr")
{
indexers = await getIndexersProwlarr()
}
else if (searchprogram == "Jackett")
{
indexers = await getIndexersJackett()
}
else if (searchprogram == "NZBHydra2")
{
indexers = await getIndexersHydra()
}
indexerCacheHelper(indexers)
return listFilter(indexers)
}
async function getIndexersJackett()
{
cachedIndexers = GM_getValue("jackettindexers", "none")
if (Date.now() - (cachedIndexers?.date || 0) < day)
{
return cachedIndexers["indexers"]
}
let params = new URLSearchParams();
params.append("apikey", GM_config.get('searchapi'));
//Invalid Category Search
params.append("Category[]", 20130)
params.append("Query", "''")
baseURL = `${GM_config.get('searchurl')}/api/v2.0/indexers/all/results`
indexerURL = `${baseURL}?${params.toString()}`
req = await fetch(indexerURL)
data = JSON.parse(req.responseText)['Indexers']
output = []
//Same keys as Prowlarr
for (i in data)
{
output.push(
{
"name": data[i]["Name"],
"id": data[i]["ID"]
})
}
GM_setValue("jacketIndexers",
{
"date": Date.now(),
"indexers": data
})
return output
}
async function getIndexersProwlarr()
{
cachedIndexers = GM_getValue("prowlarrindexers", "none")
if (Date.now() - (cachedIndexers?.date || 0) < day)
{
return cachedIndexers["indexers"]
}
let params = new URLSearchParams();
params.append("apikey", GM_config.get('searchapi'));
baseURL = `${GM_config.get('searchurl')}/api/v1/indexer`
indexerURL = `${baseURL}?${params.toString()}`
req = await fetch(indexerURL)
data = JSON.parse(req.responseText)
data = data.sort(prowlarIndexSortHelper)
GM_setValue("prowlarrindexers",
{
"date": Date.now(),
"indexers": data
})
return data
}
function prowlarIndexSortHelper(a, b)
{
if (a["priority"] > b["priority"])
{
return -1;
}
if (a["priority"] < b["priority"])
{
return 1;
}
return 0;
}
async function getIndexersHydra()
{
cachedIndexers = GM_getValue("hydraindexers", "none")
if (Date.now() - (cachedIndexers?.date || 0) < day)
{
return cachedIndexers["indexers"]
}
let params = new URLSearchParams();
params.append("apikey", GM_config.get('searchapi'));
baseURL = `${GM_config.get('searchurl')}/api/stats/indexers/`
indexerURL = `${baseURL}?${params.toString()}`
req = await fetch(indexerURL)
data = JSON.parse(req.responseText)
output = []
//Same keys as Prowlarr
for (i in data)
{
output.push(
{
"name": data[i]["indexer"],
"id": data[i]["indexer"]
})
}
GM_setValue("hydraindexers",
{
"date": Date.now(),
"indexers": output
})
return output
}
function listFilter(allIndexers)
{
let selectedIndexers = null
if (GM_config.get('listType') == "black")
{
selectedIndexers = blackListHelper(allIndexers)
}
else
{
selectedIndexers = whiteListHelper(allIndexers)
}
output = []
for (let i in allIndexers)
{
if (selectedIndexers.has(allIndexers[i]["id"]))
{
output.push(allIndexers[i])
}
}
return output
}
function indexerCacheHelper(allIndexers)
{
if (GM_config.get('indexers') == '')
{
return
}
searchprogram = GM_config.get('searchprogram')
indexerNames = GM_config.get('indexers').split(",").map((e) => e.trim().toLowerCase())
for (let j in indexerNames)
{
key = `${searchprogram}_${indexerNames[j]}`
cached = GM_getValue(key, "none")
if (cached != "none")
{
continue
}
for (let i in allIndexers)
{
if (allIndexers[i]["name"].match(new RegExp(indexerNames[j], 'i')))
{
GM_setValue(key, allIndexers[i]["id"])
}
}
}
}
function blackListHelper(allIndexers)
{
indexerID = new Set(allIndexers.map((e) => e["id"]))
indexerNames = GM_config.get('indexers').split(",").map((e) => e.trim())
for (let j in indexerNames)
{
key = `${searchprogram}_${indexerNames[j]}`
cached = GM_getValue(key, "none")
if (cached != "none")
{
indexerID.delete(cached)
}
}
return indexerID
}
function whiteListHelper(allIndexers)
{
indexerID = new Set()
indexerNames = GM_config.get('indexers').split(",").map((e) => e.trim())
for (let j in indexerNames)
{
key = `${searchprogram}_${indexerNames[j]}`
cached = GM_getValue(key, "none")
if (cached != "none")
{
indexerID.add(cached)
}
}
return indexerID
}
`
URL Processing Search
Functions Used to Produce URL for Search
`
function getSearchURLProwlarr(indexer)
{
let params = new URLSearchParams();
params.append("apikey", GM_config.get('searchapi'));
params.append("query", `${document.querySelector("#torrent-quicksearch-customsearch").value}`)
params.append("IndexerIds", indexer)
baseURL = new URL('/api/v1/search', GM_config.get('searchurl')).toString()
return `${baseURL}?${params.toString()}`
}
function getSearchURLJackett(indexer)
{
let params = new URLSearchParams();
params.append("apikey", GM_config.get('searchapi'));
params.append("Query", `${document.querySelector("#torrent-quicksearch-customsearch").value}`)
baseURL = new URL(`/api/v2.0/indexers/${indexer}/results`, GM_config.get('searchurl')).toString()
params.append("cachetime", "20")
return `${baseURL}?${params.toString()}`
}
function getSearchURLHydraNZB(indexer)
{
let params = new URLSearchParams();
params.append("apikey", GM_config.get('searchapi'));
params.append("q", `${document.querySelector("#torrent-quicksearch-customsearch").value}`)
params.append("indexers", indexer)
params.append("t", "search")
params.append("o", "xml")
params.append("cachetime", "20")
baseURL = new URL('/api', GM_config.get('searchurl')).toString()
return `${baseURL}?${params.toString()}`
}
function getSearchURLHydraTor(indexer)
{
let params = new URLSearchParams();
params.append("apikey", GM_config.get('searchapi'));
params.append("q", `${document.querySelector("#torrent-quicksearch-customsearch").value}`)
params.append("indexers", indexer)
params.append("t", "search")
params.append("o", "xml")
//hydra likes to send no data
params.append("cachetime", "20")
baseURL = new URL('/torznab/api', GM_config.get('searchurl')).toString()
return `${baseURL}?${params.toString()}`
}
`
ARR Functions
Functions to support Arr Clients
`
function getSonarrURL()
{
let params = new URLSearchParams();
params.append("apikey", GM_config.get('sonarrapi'));
baseURL = new URL('/api/v3/release/push', GM_config.get('sonarrurl')).toString()
return `${baseURL}?${params.toString()}`
}
function getRadarrURL()
{
let params = new URLSearchParams();
params.append("apikey", GM_config.get('radarrapi'));
baseURL = new URL('/api/v3/release/push', GM_config.get('radarrurl')).toString()
return `${baseURL}?${params.toString()}`
}
`
Events
These Functions Create Events to be used by script
`
function leftClickProcess(e)
{
e.preventDefault()
e.stopPropagation()
if (e.button != 0)
{
return
}
mouseState = "down"
document.addEventListener("mousemove", mouseDragProcess)
document.querySelector("#torrent-quicksearch-overlay").style.setProperty('--icon-padding', `${paddingLarge}vh`);
}
function mouseUpProcess(e)
{
e.preventDefault()
e.stopPropagation()
mouseClicksProcess()
resetMouse()
}
async function mouseClicksProcess()
{
if (mouseState == "dragged")
{
return
}
else if (Date.now() - lastClick < clickLimit)
{
return
}
else if (verifyConfig() == false)
{
GM.notification("At Minimum You Need to Set\nSearch URl\nSearch API\nSearch Program", program, searchIcon)
GM_config.open()
return
}
lastClick = Date.now()
await searchObj.toggleSearch()
}
//Reset Mouse Events
function resetMouse()
{
mouseState = "up"
document.querySelector("#torrent-quicksearch-overlay").style.setProperty('--icon-padding', `${paddingSmall}vh`);
document.removeEventListener("mousemove", mouseDragProcess)
}
function mouseDragProcess(e)
{
mouseState = "dragged"
//check mouse state on enter
if (mouseState == "up" || e.buttons == 0)
{
return
}
//poll mouse state
setInterval(() =>
{
if (mouseState == "up" || e.buttons == 0)
{
return
}
}
, 1000)
dragState = true
toggleHeight = parseInt(getComputedStyle(document.querySelector("#torrent-quicksearch-toggle")).height.replaceAll(/[^0-9.]/g, ''))
startMousePosition = parseInt(e.clientY)
offsetMousePosition = startMousePosition - (toggleHeight / 2)
viewport = (offsetMousePosition / window.innerHeight) * 100
viewport = Math.max(viewport, -20)
viewport = Math.min(viewport, 89)
document.querySelector("#torrent-quicksearch-overlay").style.top = `${viewport}vh`
}
async function sonarrFactory(releaseData)
{
sonarrEvent = async function (e)
{
e.preventDefault()
e.stopPropagation()
res = await fetch(getSonarrURL(),
{
"header": "post",
"data": JSON.stringify(releaseData)
})
if (res.status != 200)
{
GM.notification(res.responseText, program, searchIcon)
return
}
data = JSON.parse(res.responseText)
initValue = ""
finalMsg = data.reduce(
(prev, curr, index) =>
{
if (curr["rejected"] == true)
{
epNums = curr["mappedEpisodeNumbers"].length > 0 ? curr["mappedEpisodeNumbers"].join(",") : "No Episodes"
errorMsg = [`${curr["seriesTitle"]} Season ${curr["seasonNumber"]} Episodes ${epNums}`, `Status Rejected: ${curr["rejections"].join(",")}`]
return `${prev}\n\[${errorMsg.join("\n")}\]`
}
if (curr["approved"] == true)
{
acceptMsg = `Added ${curr["title"]} to client`
return `${prev}\n${acceptMsg}`
}
}, initValue)
GM.notification(finalMsg, program, searchIcon)
}
return sonarrEvent
}
async function radarrFactory(releaseData)
{
radarrEvent = async function (e)
{
e.preventDefault()
e.stopPropagation()
res = await fetch(getRadarrURL(),
{
"header": "post",
"data": JSON.stringify(releaseData)
})
if (res.status != 200)
{
GM.notification(res.responseText, program, searchIcon)
return
}
data = JSON.parse(res.responseText)
initValue = ""
finalMsg = data.reduce(
(prev, curr, index) =>
{
if (curr["rejected"] == true)
{
errorMsg = [`${curr["movieTitles"].join(",")}`, `Status Rejected: ${curr["rejections"].join(",")}`]
return `${prev}\n\[${errorMsg.join("\n")}\]`
}
if (curr["approved"] == true)
{
acceptMsg = `Added ${curr["title"]} to client`
return `${prev}\n${acceptMsg}`
}
}, initValue)
GM.notification(finalMsg, program, searchIcon)
}
return radarrEvent
}
`
Functions For Menu Events
`
function openMenu()
{
hideDisplay()
resetResultList()
resetSearchDOM()
document.querySelector("#torrent-quicksearch-overlay").style.setProperty('--icon-size', `${iconLarge}%`);
document.querySelector("#torrent-quicksearch-toggle").removeEventListener("mousedown", leftClickProcess)
document.querySelector("#torrent-quicksearch-toggle").removeEventListener("mouseup", mouseUpProcess)
searchObj.cancel()
}
function closeMenu()
{
url = GM_config.get('searchurl')
if (url.match(/htt(p|ps):\/\//) == null)
{
url = `http://${url}`
}
GM_config.set('searchurl', url)
GM_config.save()
hideDisplay()
document.querySelector("#torrent-quicksearch-toggle").addEventListener("mousedown", leftClickProcess)
document.querySelector("#torrent-quicksearch-toggle").addEventListener("mouseup", mouseUpProcess)
}
`
Globals + Main Function
`
searchIcon = ""
iconLarge = 50
iconSmall = 25
paddingSmall = 0
paddingLarge = 4
let mouseState = "up"
imdbParserFail = "Could Not Parse From Page"
AbortError = new DOMException(
'aborted!',
'AbortError')
var controller = new AbortController()
program = "Torrent Quick Search"
clickLimit = 500
lastClick = Date.now() - clickLimit
indexerSearchTimeout = 60000
day = 86400000
customSearch = false
//keep track of whether program was recently dragged
//Normalize Site Name so we don't have repeat keys in larger infoparser dict
standardNames = {
"www.imdb.com": "imdb.com",
"www.themoviedb.org": "themoviedb.org"
}
infoParser = {
"animebytes.tv":
{
"title": "h2>a[href*=series]",
"titleAttrib": "textContent"
},
"blutopia.xyz":
{
"title": "h1>a[href*=torrents\\/similar]",
"titleAttrib": "textContent",
"imdb": "div[class*=movie-details]>span>a[title=IMDB]",
"imdbAttrib": "textContent"
},
"beyond-hd.me":
{
"title": "h1[class=movie-heading]",
"titleAttrib": "textContent",
"imdb": "ul[class*=movie-details]>li>span>a[href*=imdb]",
"imdbAttrib": "href"
},
"imdb.com":
{
"title": "h1",
"titleAttrib": "textContent",
},
"themoviedb.org":
{
"title": "h2",
"titleAttrib": "textContent",
},
}
siteParser = getParser()
//Main Function
function main()
{
createMainDOM()
GM_config.init(
{
'id': 'torrent-quick-search',
'title': 'Torrent Quick Search Settings', // Panel Title
'fields':
{
'tmdbapi':
{
'label': 'TMDB API Key',
'type': 'text',
'title': 'TMDB Key For TMDB/IMDB conversion'
},
'searchurl':
{
'label': 'Search URL',
'section': ['Search'],
'type': 'text',
'title': 'Base URl for search program'
},
'searchapi':
{
'label': 'API Key',
'type': 'text',
'title': 'API key for search program'
},
'searchprogram':
{
'label': 'Search Program',
'type': 'select',
'options': ['Prowlarr', 'Jackett', "NZBHydra2"],
'title': 'Which search program'
},
'sitefilter':
{
'label': 'Filter Current Site',
'type': 'radio',
'options': ['true', 'false'],
'title': 'Should Results From Current Site be Filtered Out'
},
'indexers':
{
'label': 'Indexers',
'section': ['Indexers'],
'type': 'text',
'title': 'Comma Seperated List of Indexers Names'
},
'listType':
{
'type': 'radio',
'options': ['black', 'white'],
'label': 'Indexers ListType',
'title': 'Use White list if you want to manually approve indexers\nUse Black list if you want to use all Indexers, and just have few or none to disable'
},
'radarrurl':
{
'label': 'Radarr URL',
'section': ['Arr Clients'],
'type': 'text',
'title': 'URL for Radarr',
},
'radarrapi':
{
'label': 'Radarr ApiKey ',
'type': 'text',
'title': 'APIKey for Radarr',
},
'sonarrurl':
{
'label': 'Sonarr URL',
'type': 'text',
'title': 'URL for Sonarr',
},
'sonarrapi':
{
'label': 'Sonarr ApiKey ',
'type': 'text',
'title': 'APIKey for Sonarr',
},
'fontsize':
{
'label': 'Font Size',
'section': ['GUI'],
'type': 'int',
'title': 'fontsize',
'default': 12
},
},
'events':
{
'open': openMenu,
'close': closeMenu
},
});
GM.registerMenuCommand('Torrent Quick Search Settings', function ()
{
GM_config.open();
});
}
main()