mirror of
https://github.com/knightcrawler-stremio/knightcrawler.git
synced 2024-12-20 03:29:51 +00:00
105 lines
4.0 KiB
JavaScript
105 lines
4.0 KiB
JavaScript
const needle = require('needle');
|
|
const { encode } = require('magnet-uri');
|
|
const RealDebridClient = require('real-debrid-api');
|
|
const { cacheWrapUnrestricted } = require('../lib/cache');
|
|
|
|
const REAL_DEBRID_API_URL = 'https://api.real-debrid.com/rest/1.0';
|
|
const REAL_DEBRID_UNRESTRICTER_URL = 'http://localhost:7050';
|
|
|
|
async function applyMoch(streams, token) {
|
|
const streamMapping = streams.reduce((map, stream) => (map[stream.infoHash] = stream, map), {});
|
|
const hashes = streams.map(stream => stream.infoHash);
|
|
const available = await _instantAvailability(hashes, token).catch(() => undefined);
|
|
if (available) {
|
|
Object.entries(available)
|
|
.filter(([key, value]) => isCachedFileAvailable(streamMapping[key.toLowerCase()], value))
|
|
.map(([key]) => key.toLowerCase())
|
|
.map(cachedInfoHash => streams.find(stream => stream.infoHash === cachedInfoHash))
|
|
.forEach(stream => {
|
|
stream.name = `[RD Cached]\n${stream.name}`;
|
|
stream.url = `${REAL_DEBRID_UNRESTRICTER_URL}/realdebrid/${token}/${stream.infoHash}/${stream.fileIdx}`;
|
|
delete stream.infoHash;
|
|
delete stream.fileIndex;
|
|
})
|
|
}
|
|
|
|
return streams;
|
|
}
|
|
|
|
async function unrestrict(apiKey, infoHash, fileIndex) {
|
|
if (!apiKey || !infoHash) {
|
|
return Promise.reject("No valid parameters passed");
|
|
}
|
|
const key = `${apiKey}_${infoHash}_${fileIndex}`;
|
|
return cacheWrapUnrestricted(key, () => _unrestrict(apiKey, infoHash, fileIndex))
|
|
}
|
|
|
|
async function _unrestrict(apiKey, infoHash, fileIndex) {
|
|
if (!apiKey || !infoHash) {
|
|
return Promise.reject("No valid parameters passed");
|
|
}
|
|
console.log(`Unrestricting ${infoHash} [${fileIndex}]`);
|
|
const RD = new RealDebridClient(apiKey);
|
|
const torrentId = await _createOrFindTorrentId(RD, infoHash);
|
|
console.log(`Retrieved torrentId: ${torrentId}`);
|
|
if (torrentId) {
|
|
const info = await RD.torrents.info(torrentId);
|
|
const downloadFile = info.files.find(file => file.id === fileIndex + 1);
|
|
const selectedFiles = info.files.filter(file => file.selected);
|
|
const fileLink = info.links.length === 1
|
|
? info.links[0]
|
|
: info.links[selectedFiles.indexOf(downloadFile)];
|
|
const unrestrictedLink = fileLink && await RD.unrestrict.link(fileLink);
|
|
|
|
if (unrestrictedLink) {
|
|
console.log(`Unrestricted ${infoHash} [${fileIndex}] to ${unrestrictedLink.download}`);
|
|
return Promise.resolve(unrestrictedLink.download);
|
|
}
|
|
return Promise.reject("No available links found");
|
|
}
|
|
return Promise.reject("Failed adding torrent");
|
|
}
|
|
|
|
async function _createOrFindTorrentId(RD, infoHash) {
|
|
return RD.torrents.get(0, 1)
|
|
.then(torrents => torrents.find(torrent => torrent.hash.toLowerCase() === infoHash))
|
|
.then(torrent => torrent && torrent.id || Promise.reject('No recent torrent found'))
|
|
.catch((error) => RD.torrents.addMagnet(encode({ infoHash }))
|
|
.then(response => RD.torrents.selectFiles(response.id)
|
|
.then((() => response.id))))
|
|
.catch(error => {
|
|
console.warn('Failed RealDebrid torrent retrieval', error);
|
|
return undefined;
|
|
});
|
|
}
|
|
|
|
function isCachedFileAvailable(stream, hosterResults) {
|
|
if (!hosterResults || Array.isArray(hosterResults)) {
|
|
return false;
|
|
}
|
|
return !!Object.values(hosterResults)
|
|
.reduce((a, b) => a.concat(b), [])
|
|
.filter(cached => isNaN(stream.fileIdx) && Object.keys(cached).length || cached[stream.fileIdx + 1])
|
|
.length;
|
|
}
|
|
|
|
async function _instantAvailability(hashes, token) {
|
|
const endpoint = `/torrents/instantAvailability/${hashes.join('/')}`;
|
|
return _request(endpoint, { token });
|
|
}
|
|
|
|
async function _request(endpoint, config) {
|
|
const url = REAL_DEBRID_API_URL + endpoint;
|
|
const method = config.method || 'get';
|
|
const headers = { 'Authorization': 'Bearer ' + config.token };
|
|
|
|
return needle(method, url, { headers })
|
|
.then(response => {
|
|
if (response.body && typeof response.body === 'object') {
|
|
return response.body;
|
|
}
|
|
return Promise.reject('No response body');
|
|
})
|
|
}
|
|
|
|
module.exports = { applyMoch, unrestrict }; |