mirror of
https://github.com/knightcrawler-stremio/knightcrawler.git
synced 2024-12-20 03:29:51 +00:00
139 lines
4.8 KiB
JavaScript
139 lines
4.8 KiB
JavaScript
const { torrentFiles } = require('../lib/torrent');
|
|
const { escapeTitle, getMetadata, getImdbId } = require('../lib/metadata');
|
|
const { parse } = require('parse-torrent-title');
|
|
const { Type } = require('./types');
|
|
|
|
const MIN_SIZE = 20 * 1024 * 1024; // 20 MB
|
|
|
|
async function parseTorrentFiles(torrent, imdbId, kitsuId) {
|
|
const parsedTorrentName = parse(torrent.title);
|
|
parsedTorrentName.hasMovies = parsedTorrentName.complete || !!torrent.title.match(/movies?(?:\W|$)/);
|
|
const metadata = await getMetadata(kitsuId || imdbId, torrent.type || Type.MOVIE).catch(() => undefined);
|
|
|
|
if (metadata && metadata.type !== torrent.type && torrent.type !== Type.ANIME) {
|
|
throw new Error(`Mismatching entry type for ${torrent.name}: ${torrent.type}!=${metadata.type}`);
|
|
}
|
|
|
|
if (torrent.type === Type.MOVIE) {
|
|
if (parsedTorrentName.complete) {
|
|
return torrentFiles(torrent)
|
|
.then(files => files.filter(file => file.size > MIN_SIZE))
|
|
.then(files => Promise.all(files
|
|
.map((file) => findMovieImdbId(file.name)
|
|
.then((newImdbId) => ({
|
|
infoHash: torrent.infoHash,
|
|
fileIndex: file.fileIndex,
|
|
title: file.name,
|
|
size: file.size,
|
|
imdbId: newImdbId,
|
|
})))))
|
|
.catch(error => {
|
|
console.log(`Failed getting files for ${torrent.title}`, error.message);
|
|
return [];
|
|
});
|
|
}
|
|
|
|
return [ {
|
|
infoHash: torrent.infoHash,
|
|
title: torrent.title,
|
|
size: torrent.size,
|
|
imdbId: imdbId || metadata && metadata.imdb_id,
|
|
kitsuId: kitsuId || metadata && metadata.kitsu_id
|
|
} ];
|
|
}
|
|
|
|
return getSeriesFiles(torrent, parsedTorrentName)
|
|
.then((files) => files
|
|
.filter((file) => file.size > MIN_SIZE)
|
|
.map((file) => parseSeriesFile(file, parsedTorrentName)))
|
|
.then((files) => decomposeAbsoluteEpisodes(files, metadata))
|
|
.then((files) => Promise.all(files.map(file => file.isMovie
|
|
? mapSeriesMovie(file, torrent.infoHash)
|
|
: mapSeriesEpisode(file, torrent.infoHash, imdbId))))
|
|
.then((files) => files.reduce((a, b) => a.concat(b), []))
|
|
.catch((error) => {
|
|
console.log(`Failed getting files for ${torrent.title}`, error.message);
|
|
return [];
|
|
});
|
|
}
|
|
|
|
async function getSeriesFiles(torrent, parsedTorrentName) {
|
|
if (parsedTorrentName.episode || parsedTorrentName.date) {
|
|
return [ {
|
|
name: torrent.title,
|
|
path: torrent.title,
|
|
size: torrent.size
|
|
} ];
|
|
}
|
|
|
|
return torrentFiles(torrent);
|
|
}
|
|
|
|
async function mapSeriesEpisode(file, infoHash, imdbId) {
|
|
if (!file.episodes) {
|
|
return Promise.resolve([]);
|
|
}
|
|
return Promise.resolve(file.episodes.map(episode => ({
|
|
infoHash: infoHash,
|
|
fileIndex: file.fileIndex,
|
|
title: file.path || file.name,
|
|
size: file.size,
|
|
imdbId: imdbId,
|
|
imdbSeason: file.season,
|
|
imdbEpisode: episode
|
|
})))
|
|
}
|
|
|
|
async function mapSeriesMovie(file, infoHash) {
|
|
return findMovieImdbId(file).then((imdbId) => [ {
|
|
infoHash: infoHash,
|
|
fileIndex: file.fileIndex,
|
|
title: file.name,
|
|
size: file.size,
|
|
imdbId: imdbId
|
|
} ])
|
|
}
|
|
|
|
function parseSeriesFile(file, parsedTorrentName) {
|
|
const fileInfo = parse(file.name);
|
|
// the episode may be in a folder containing season number
|
|
if (!fileInfo.season && parsedTorrentName.season) {
|
|
fileInfo.season = parsedTorrentName.season;
|
|
} else if (!fileInfo.season && file.path.includes('/')) {
|
|
const folders = file.path.split('/');
|
|
const pathInfo = parse(folders[folders.length - 2]);
|
|
fileInfo.season = pathInfo.season;
|
|
}
|
|
fileInfo.isMovie = parsedTorrentName.hasMovies && !fileInfo.season &&
|
|
(!fileInfo.episodes || !!fileInfo.year || !!file.name.match(/\b(?:\d+[ .]movie|movie[ .]\d+)\b/i));
|
|
|
|
return { ...file, ...fileInfo };
|
|
}
|
|
|
|
function findMovieImdbId(title) {
|
|
const parsedTitle = typeof title === 'string' ? parse(title) : title;
|
|
const searchQuery = {
|
|
name: escapeTitle(parsedTitle.title).toLowerCase(),
|
|
year: parsedTitle.year,
|
|
type: Type.MOVIE
|
|
};
|
|
return getImdbId(searchQuery).catch((error) => undefined);
|
|
}
|
|
|
|
async function decomposeAbsoluteEpisodes(files, metadata) {
|
|
if (files.every((file) => !file.episodes || file.episodes.every((ep) => ep < 100))) {
|
|
return files; // nothing to decompose
|
|
}
|
|
|
|
// decompose if season is inside path, but individual files are concatenated ex. 101 (S01E01)
|
|
files
|
|
.filter(file => file.season && metadata.episodeCount[file.season] < 100)
|
|
.filter(file => file.episodes && file.episodes.every(ep => ep / 100 === file.season))
|
|
.forEach(file => file.episodes = file.episodes.map(ep => ep % 100));
|
|
// decompose if no season info is available, but individual files are concatenated ex. 101 (S01E01)
|
|
// based on total episodes count per season
|
|
|
|
return files;
|
|
}
|
|
|
|
module.exports = { parseTorrentFiles }; |