143 lines
5.0 KiB
JavaScript
143 lines
5.0 KiB
JavaScript
const { addonBuilder } = require('stremio-addon-sdk');
|
|
const { manifest } = require('./lib/manifest');
|
|
const { cacheWrapStream } = require('./lib/cache');
|
|
const { toStreamInfo, sanitizeStreamInfo } = require('./lib/streamInfo');
|
|
const repository = require('./lib/repository');
|
|
const realdebrid = require('./moch/realdebrid');
|
|
|
|
const CACHE_MAX_AGE = process.env.CACHE_MAX_AGE || 4 * 60 * 60; // 4 hours in seconds
|
|
const CACHE_MAX_AGE_EMPTY = 30 * 60; // 30 minutes
|
|
const STALE_REVALIDATE_AGE = 4 * 60 * 60; // 4 hours
|
|
const STALE_ERROR_AGE = 7 * 24 * 60 * 60; // 7 days
|
|
|
|
const MOCHS = {
|
|
'realdebrid': realdebrid
|
|
};
|
|
|
|
const builder = new addonBuilder(manifest());
|
|
|
|
builder.defineStreamHandler((args) => {
|
|
if (!args.id.match(/tt\d+/i) && !args.id.match(/kitsu:\d+/i)) {
|
|
return Promise.resolve({ streams: [] });
|
|
}
|
|
|
|
const handlers = {
|
|
series: () => seriesRecordsHandler(args).then(records => records.map(record => toStreamInfo(record))),
|
|
movie: () => movieRecordsHandler(args).then(records => records.map(record => toStreamInfo(record))),
|
|
fallback: () => Promise.reject('not supported type')
|
|
};
|
|
|
|
return cacheWrapStream(args.id, (handlers[args.type] || handlers.fallback))
|
|
.then(streams => filterStreamByProvider(streams, args.extra.providers))
|
|
.then(streams => filterStreamsBySeeders(streams))
|
|
.then(streams => sortStreamsByVideoQuality(streams))
|
|
.then(streams => applyMochs(streams, args.extra))
|
|
.then(streams => streams.map(stream => sanitizeStreamInfo(stream)))
|
|
.then(streams => ({
|
|
streams: streams,
|
|
cacheMaxAge: streams.length ? CACHE_MAX_AGE : CACHE_MAX_AGE_EMPTY,
|
|
staleRevalidate: STALE_REVALIDATE_AGE,
|
|
staleError: STALE_ERROR_AGE
|
|
}))
|
|
.catch(error => {
|
|
console.log(`Failed request ${args.id}: ${error}`);
|
|
throw error;
|
|
});
|
|
});
|
|
|
|
async function seriesRecordsHandler(args) {
|
|
if (args.id.match(/tt\d+/)) {
|
|
const parts = args.id.split(':');
|
|
const imdbId = parts[0];
|
|
const season = parseInt(parts[1], 10);
|
|
const episode = parseInt(parts[2], 10);
|
|
return repository.getImdbIdSeriesEntries(imdbId, season, episode);
|
|
} else if (args.id.match(/kitsu:\d+/i)) {
|
|
const parts = args.id.split(':');
|
|
const kitsuId = parts[1];
|
|
const episode = parseInt(parts[2], 10);
|
|
return repository.getKitsuIdSeriesEntries(kitsuId, episode);
|
|
}
|
|
return Promise.reject(`Unsupported id type: ${args.id}`);
|
|
}
|
|
|
|
async function movieRecordsHandler(args) {
|
|
if (args.id.match(/tt\d+/)) {
|
|
return repository.getImdbIdMovieEntries(args.id);
|
|
} else if (args.id.match(/kitsu:\d+/i)) {
|
|
return repository.getKitsuIdMovieEntries(args.id.replace('kitsu:', ''));
|
|
}
|
|
return Promise.reject(`Unsupported id type: ${args.id}`);
|
|
}
|
|
|
|
function filterStreamByProvider(streams, providers) {
|
|
if (!providers || !providers.length) {
|
|
return streams;
|
|
}
|
|
return streams.filter(stream => providers.includes(stream.name.split('\n')[1].toLowerCase()))
|
|
}
|
|
|
|
const HEALTHY_SEEDERS = 5;
|
|
const SEEDED_SEEDERS = 1;
|
|
const MIN_HEALTHY_COUNT = 10;
|
|
const MAX_UNHEALTHY_COUNT = 5;
|
|
|
|
function filterStreamsBySeeders(streams) {
|
|
const sortedStreams = streams
|
|
.sort((a, b) => b.filters.seeders - a.filters.seeders || b.filters.uploadDate - a.filters.uploadDate);
|
|
const healthy = sortedStreams.filter(stream => stream.filters.seeders >= HEALTHY_SEEDERS);
|
|
const seeded = sortedStreams.filter(stream => stream.filters.seeders >= SEEDED_SEEDERS);
|
|
|
|
if (healthy.length >= MIN_HEALTHY_COUNT) {
|
|
return healthy;
|
|
} else if (seeded.length >= MAX_UNHEALTHY_COUNT) {
|
|
return seeded.slice(0, MIN_HEALTHY_COUNT);
|
|
}
|
|
return sortedStreams.slice(0, MAX_UNHEALTHY_COUNT);
|
|
}
|
|
|
|
function sortStreamsByVideoQuality(streams) {
|
|
const qualityMap = streams
|
|
.reduce((map, stream) => {
|
|
const quality = stream.filters.quality;
|
|
map[quality] = (map[quality] || []).concat(stream);
|
|
return map;
|
|
}, {});
|
|
const sortedQualities = Object.keys(qualityMap)
|
|
.sort((a, b) => {
|
|
const aQuality = a === '4k' ? '2160p' : a;
|
|
const bQuality = b === '4k' ? '2160p' : b;
|
|
const aResolution = aQuality && aQuality.match(/\d+p/) && parseInt(aQuality, 10);
|
|
const bResolution = bQuality && bQuality.match(/\d+p/) && parseInt(bQuality, 10);
|
|
if (aResolution && bResolution) {
|
|
return bResolution - aResolution; // higher resolution first;
|
|
} else if (aResolution) {
|
|
return -1;
|
|
} else if (bResolution) {
|
|
return 1;
|
|
}
|
|
return a < b ? -1 : b < a ? 1 : 0;
|
|
});
|
|
return sortedQualities
|
|
.map(quality => qualityMap[quality])
|
|
.reduce((a, b) => a.concat(b), []);
|
|
}
|
|
|
|
function applyMochs(streams, config) {
|
|
if (!streams || !streams.length) {
|
|
return streams;
|
|
}
|
|
|
|
return Object.keys(config)
|
|
.filter(configKey => MOCHS[configKey])
|
|
.reduce(async (streams, moch) => {
|
|
return await MOCHS[moch].applyMoch(streams, config[moch])
|
|
.catch(error => {
|
|
console.warn(error);
|
|
return streams;
|
|
});
|
|
}, streams);
|
|
}
|
|
|
|
module.exports = builder.getInterface();
|