mirror of
https://github.com/knightcrawler-stremio/knightcrawler.git
synced 2024-12-20 03:29:51 +00:00
187 lines
7.0 KiB
JavaScript
187 lines
7.0 KiB
JavaScript
import OffcloudClient from 'offcloud-api';
|
|
import magnet from 'magnet-uri';
|
|
import { Type } from '../lib/types.js';
|
|
import { isVideo } from '../lib/extension.js';
|
|
import StaticResponse from './static.js';
|
|
import { getMagnetLink } from '../lib/magnetHelper.js';
|
|
import { chunkArray, BadTokenError, sameFilename } from './mochHelper.js';
|
|
|
|
const KEY = 'offcloud';
|
|
|
|
export async function getCachedStreams(streams, apiKey) {
|
|
const options = await getDefaultOptions();
|
|
const OC = new OffcloudClient(apiKey, options);
|
|
const hashBatches = chunkArray(streams.map(stream => stream.infoHash), 100);
|
|
const available = await Promise.all(hashBatches.map(hashes => OC.instant.cache(hashes)))
|
|
.then(results => results.map(result => result.cachedItems))
|
|
.then(results => results.reduce((all, result) => all.concat(result), []))
|
|
.catch(error => {
|
|
if (toCommonError(error)) {
|
|
return Promise.reject(error);
|
|
}
|
|
console.warn('Failed Offcloud cached torrent availability request:', error);
|
|
return undefined;
|
|
});
|
|
return available && streams
|
|
.reduce((mochStreams, stream) => {
|
|
const isCached = available.includes(stream.infoHash);
|
|
const streamTitleParts = stream.title.replace(/\n👤.*/s, '').split('\n');
|
|
const fileName = streamTitleParts[streamTitleParts.length - 1];
|
|
const fileIndex = streamTitleParts.length === 2 ? stream.fileIdx : null;
|
|
const encodedFileName = encodeURIComponent(fileName);
|
|
mochStreams[stream.infoHash] = {
|
|
url: `${apiKey}/${stream.infoHash}/${encodedFileName}/${fileIndex}`,
|
|
cached: isCached
|
|
};
|
|
return mochStreams;
|
|
}, {})
|
|
}
|
|
|
|
export async function getCatalog(apiKey, offset = 0) {
|
|
if (offset > 0) {
|
|
return [];
|
|
}
|
|
const options = await getDefaultOptions();
|
|
const OC = new OffcloudClient(apiKey, options);
|
|
return OC.cloud.history()
|
|
.then(torrents => torrents)
|
|
.then(torrents => (torrents || [])
|
|
.map(torrent => ({
|
|
id: `${KEY}:${torrent.requestId}`,
|
|
type: Type.OTHER,
|
|
name: torrent.fileName
|
|
})));
|
|
}
|
|
|
|
export async function getItemMeta(itemId, apiKey, ip) {
|
|
const options = await getDefaultOptions(ip);
|
|
const OC = new OffcloudClient(apiKey, options);
|
|
const torrents = await OC.cloud.history();
|
|
const torrent = torrents.find(torrent => torrent.requestId === itemId)
|
|
const infoHash = torrent && magnet.decode(torrent.originalLink).infoHash
|
|
const createDate = torrent ? new Date(torrent.createdOn) : new Date();
|
|
return _getFileUrls(OC, torrent)
|
|
.then(files => ({
|
|
id: `${KEY}:${itemId}`,
|
|
type: Type.OTHER,
|
|
name: torrent.name,
|
|
infoHash: infoHash,
|
|
videos: files
|
|
.filter(file => isVideo(file))
|
|
.map((file, index) => ({
|
|
id: `${KEY}:${itemId}:${index}`,
|
|
title: file.split('/').pop(),
|
|
released: new Date(createDate.getTime() - index).toISOString(),
|
|
streams: [{ url: file }]
|
|
}))
|
|
}))
|
|
}
|
|
|
|
export async function resolve({ ip, apiKey, infoHash, cachedEntryInfo, fileIndex }) {
|
|
console.log(`Unrestricting Offcloud ${infoHash} [${fileIndex}]`);
|
|
const options = await getDefaultOptions(ip);
|
|
const OC = new OffcloudClient(apiKey, options);
|
|
|
|
return _resolve(OC, infoHash, cachedEntryInfo, fileIndex)
|
|
.catch(error => {
|
|
if (errorExpiredSubscriptionError(error)) {
|
|
console.log(`Access denied to Offcloud ${infoHash} [${fileIndex}]`);
|
|
return StaticResponse.FAILED_ACCESS;
|
|
}
|
|
return Promise.reject(`Failed Offcloud adding torrent ${JSON.stringify(error)}`);
|
|
});
|
|
}
|
|
|
|
async function _resolve(OC, infoHash, cachedEntryInfo, fileIndex) {
|
|
const torrent = await _createOrFindTorrent(OC, infoHash)
|
|
.then(info => info.requestId ? OC.cloud.status(info.requestId) : Promise.resolve(info))
|
|
.then(info => info.status || info);
|
|
if (torrent && statusReady(torrent)) {
|
|
return _unrestrictLink(OC, infoHash, torrent, cachedEntryInfo, fileIndex);
|
|
} else if (torrent && statusDownloading(torrent)) {
|
|
console.log(`Downloading to Offcloud ${infoHash} [${fileIndex}]...`);
|
|
return StaticResponse.DOWNLOADING;
|
|
} else if (torrent && statusError(torrent)) {
|
|
console.log(`Retry failed download in Offcloud ${infoHash} [${fileIndex}]...`);
|
|
return _retryCreateTorrent(OC, infoHash, cachedEntryInfo, fileIndex);
|
|
}
|
|
|
|
return Promise.reject(`Failed Offcloud adding torrent ${JSON.stringify(torrent)}`);
|
|
}
|
|
|
|
async function _createOrFindTorrent(OC, infoHash) {
|
|
return _findTorrent(OC, infoHash)
|
|
.catch(() => _createTorrent(OC, infoHash));
|
|
}
|
|
|
|
async function _findTorrent(OC, infoHash) {
|
|
const torrents = await OC.cloud.history();
|
|
const foundTorrents = torrents.filter(torrent => torrent.originalLink.toLowerCase().includes(infoHash));
|
|
const nonFailedTorrent = foundTorrents.find(torrent => !statusError(torrent));
|
|
const foundTorrent = nonFailedTorrent || foundTorrents[0];
|
|
return foundTorrent || Promise.reject('No recent torrent found');
|
|
}
|
|
|
|
async function _createTorrent(OC, infoHash) {
|
|
const magnetLink = await getMagnetLink(infoHash);
|
|
return OC.cloud.download(magnetLink)
|
|
}
|
|
|
|
async function _retryCreateTorrent(OC, infoHash, cachedEntryInfo, fileIndex) {
|
|
const newTorrent = await _createTorrent(OC, infoHash);
|
|
return newTorrent && statusReady(newTorrent.status)
|
|
? _unrestrictLink(OC, infoHash, newTorrent, cachedEntryInfo, fileIndex)
|
|
: StaticResponse.FAILED_DOWNLOAD;
|
|
}
|
|
|
|
async function _unrestrictLink(OC, infoHash, torrent, cachedEntryInfo, fileIndex) {
|
|
const targetFileName = decodeURIComponent(cachedEntryInfo);
|
|
const files = await _getFileUrls(OC, torrent)
|
|
const targetFile = files.find(file => sameFilename(targetFileName, file.split('/').pop()))
|
|
|| files.find(file => isVideo(file))
|
|
|| files.pop();
|
|
|
|
if (!targetFile) {
|
|
return Promise.reject(`No Offcloud links found for index ${fileIndex} in: ${JSON.stringify(torrent)}`);
|
|
}
|
|
console.log(`Unrestricted Offcloud ${infoHash} [${fileIndex}] to ${targetFile}`);
|
|
return targetFile;
|
|
}
|
|
|
|
async function _getFileUrls(OC, torrent) {
|
|
return OC.cloud.explore(torrent.requestId)
|
|
.catch(error => {
|
|
if (error === 'Bad archive') {
|
|
return [`https://${torrent.server}.offcloud.com/cloud/download/${torrent.requestId}/${torrent.fileName}`];
|
|
}
|
|
throw error;
|
|
})
|
|
}
|
|
|
|
async function getDefaultOptions(ip) {
|
|
return { ip, timeout: 10000 };
|
|
}
|
|
|
|
export function toCommonError(error) {
|
|
if (error?.error === 'NOAUTH' || error?.message?.startsWith('Cannot read property')) {
|
|
return BadTokenError;
|
|
}
|
|
return undefined;
|
|
}
|
|
|
|
function statusDownloading(torrent) {
|
|
return ['downloading', 'created'].includes(torrent.status);
|
|
}
|
|
|
|
function statusError(torrent) {
|
|
return ['error', 'canceled'].includes(torrent.status);
|
|
}
|
|
|
|
function statusReady(torrent) {
|
|
return torrent.status === 'downloaded';
|
|
}
|
|
|
|
function errorExpiredSubscriptionError(error) {
|
|
return error?.includes && (error.includes('not_available') || error.includes('NOAUTH') || error.includes('premium membership'));
|
|
}
|