mirror of
https://github.com/knightcrawler-stremio/knightcrawler.git
synced 2024-12-20 03:29:51 +00:00
92 lines
3.3 KiB
JavaScript
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();
|