Files
knightcrawler/addon/addon.js
2020-04-05 21:42:44 +02:00

92 lines
3.3 KiB
JavaScript

const { addonBuilder } = require('stremio-addon-sdk');
const { Type } = require('./lib/types');
const { manifest, Providers } = require('./lib/manifest');
const { cacheWrapStream } = require('./lib/cache');
const { toStreamInfo } = require('./lib/streamInfo');
const repository = require('./lib/repository');
const applySorting = require('./lib/sort');
const applyMochs = require('./moch/moch');
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 defaultProviders = Providers.map(provider => provider.toLowerCase());
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: [] });
}
return cacheWrapStream(args.id, () => streamHandler(args)
.then(records => records
.sort((a, b) => b.torrent.seeders - a.torrent.seeders || b.torrent.uploadDate - a.torrent.uploadDate)
.map(record => toStreamInfo(record))))
.then(streams => filterByProvider(streams, args.extra.providers || defaultProviders))
.then(streams => applySorting(streams, args.extra))
.then(streams => applyMochs(streams, args.extra))
.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 streamHandler(args) {
if (args.type === Type.MOVIE) {
return movieRecordsHandler(args);
} else if (args.type === Type.SERIES) {
return seriesRecordsHandler(args);
}
return Promise.reject('not supported type');
}
async function seriesRecordsHandler(args) {
if (args.id.match(/tt\d+/)) {
const parts = args.id.split(':');
const imdbId = parts[0];
const season = parts[1] ? parseInt(parts[1], 10) : 1;
const episode = parts[2] ? parseInt(parts[2], 10) : 1;
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 = parts[2] ? parseInt(parts[2], 10) : 1;
return repository.getKitsuIdSeriesEntries(kitsuId, episode);
}
return Promise.reject(`Unsupported id type: ${args.id}`);
}
async function movieRecordsHandler(args) {
if (args.id.match(/tt\d+/)) {
const parts = args.id.split(':');
const imdbId = parts[0];
return repository.getImdbIdMovieEntries(imdbId);
} else if (args.id.match(/kitsu:\d+/i)) {
const parts = args.id.split(':');
const kitsuId = parts[1];
return repository.getKitsuIdMovieEntries(kitsuId);
}
return Promise.reject(`Unsupported id type: ${args.id}`);
}
function filterByProvider(streams, providers) {
if (!providers || !providers.length) {
return streams;
}
return streams.filter(stream => {
const parts = stream.title.split('\n');
const provider = parts[parts.length - 2].match(/\w+$/)[0];
return providers.includes(provider.toLowerCase());
})
}
module.exports = builder.getInterface();