Compare commits
18 Commits
ffmpeg_pul
...
v1.1.10
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
56afce02ea | ||
|
|
bc4697061e | ||
|
|
55ffe85f77 | ||
|
|
ec8c2c7716 | ||
|
|
714a528f8b | ||
|
|
8314d91bd7 | ||
|
|
5bd31f9e0b | ||
|
|
95a93930f3 | ||
|
|
4a9e1d0410 | ||
|
|
1eacd0a5ca | ||
|
|
3c32726745 | ||
|
|
42ae0ae1fb | ||
|
|
e4b3871919 | ||
|
|
58e4a557e2 | ||
|
|
8371d68113 | ||
|
|
b7d496fc9d | ||
|
|
14260d04b3 | ||
|
|
3d46b65d67 |
1
LICENSE
1
LICENSE
@@ -1,4 +1,5 @@
|
|||||||
Copyright (c) 2015 Roel van Uden
|
Copyright (c) 2015 Roel van Uden
|
||||||
|
Copyright (c) 2016 Manoel <Godzil> Trapier
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
of this software and associated documentation files (the "Software"), to
|
of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
24
README.md
24
README.md
@@ -1,6 +1,6 @@
|
|||||||
# CrunchyRoll.js
|
# Crunchy.js: a fork of Deathspike/CrunchyRoll.js
|
||||||
|
|
||||||
*CrunchyRoll.js* is capable of downloading *anime* episodes from the popular *CrunchyRoll* streaming service. An episode is stored in the original video format (often H.264 in a MP4 container) and the configured subtitle format (ASS or SRT).The two output files are then merged into a single MKV file.
|
*Crunchy.js* is capable of downloading *anime* episodes from the popular *CrunchyRoll* streaming service. An episode is stored in the original video format (often H.264 in a MP4 container) and the configured subtitle format (ASS or SRT).The two output files are then merged into a single MKV file.
|
||||||
|
|
||||||
## Motivation
|
## Motivation
|
||||||
|
|
||||||
@@ -10,6 +10,8 @@
|
|||||||
|
|
||||||
This application is not endorsed or affliated with *CrunchyRoll*. The usage of this application enables episodes to be downloaded for offline convenience which may be forbidden by law in your country. Usage of this application may also cause a violation of the agreed *Terms of Service* between you and the stream provider. A tool is not responsible for your actions; please make an informed decision prior to using this application.
|
This application is not endorsed or affliated with *CrunchyRoll*. The usage of this application enables episodes to be downloaded for offline convenience which may be forbidden by law in your country. Usage of this application may also cause a violation of the agreed *Terms of Service* between you and the stream provider. A tool is not responsible for your actions; please make an informed decision prior to using this application.
|
||||||
|
|
||||||
|
**PLEASE USE THIS TOOL ONLY IF YOU HAVE A PREMIUM ACCOUNT**
|
||||||
|
|
||||||
## Configuration
|
## Configuration
|
||||||
|
|
||||||
It is recommended to enable authentication (`-p` and `-u`) so your account permissions and settings are available for use. It is not possible to download non-free material without an account and premium subscription. Furthermore, the default account settings are used when downloading. If you want the highest quality videos, configure these preferences at https://www.crunchyroll.com/acct/?action=video.
|
It is recommended to enable authentication (`-p` and `-u`) so your account permissions and settings are available for use. It is not possible to download non-free material without an account and premium subscription. Furthermore, the default account settings are used when downloading. If you want the highest quality videos, configure these preferences at https://www.crunchyroll.com/acct/?action=video.
|
||||||
@@ -28,28 +30,28 @@ Use the applicable instructions to install. Is your operating system not listed?
|
|||||||
|
|
||||||
1. Run in *Terminal*: `sudo apt-get install nodejs npm mkvtoolnix rtmpdump ffmpeg`
|
1. Run in *Terminal*: `sudo apt-get install nodejs npm mkvtoolnix rtmpdump ffmpeg`
|
||||||
2. Run in *Terminal*: `sudo ln -s /usr/bin/nodejs /usr/bin/node`
|
2. Run in *Terminal*: `sudo ln -s /usr/bin/nodejs /usr/bin/node`
|
||||||
3. Run in *Terminal*: `sudo npm install -g crunchyroll`
|
3. Run in *Terminal*: `sudo npm install -g crunchy`
|
||||||
|
|
||||||
### Mac OS X
|
### Mac OS X
|
||||||
|
|
||||||
1. Install *Homebrew* following the instructions at http://brew.sh/
|
1. Install *Homebrew* following the instructions at http://brew.sh/
|
||||||
2. Run in *Terminal*: `brew install node mkvtoolnix rtmpdump ffmpeg`
|
2. Run in *Terminal*: `brew install node mkvtoolnix rtmpdump ffmpeg`
|
||||||
3. Run in *Terminal*: `npm install -g crunchyroll`
|
3. Run in *Terminal*: `npm install -g crunchy`
|
||||||
|
|
||||||
### Windows
|
### Windows
|
||||||
|
|
||||||
1. Install *NodeJS* following the instructions at http://nodejs.org/
|
1. Install *NodeJS* following the instructions at http://nodejs.org/
|
||||||
3. Run in *Command Prompt*: `npm install -g crunchyroll`
|
3. Run in *Command Prompt*: `npm install -g crunchy`
|
||||||
|
|
||||||
## Instructions
|
## Instructions
|
||||||
|
|
||||||
Use the applicable instructions for the interface of your choice (currently limited to command-line).
|
Use the applicable instructions for the interface of your choice (currently limited to command-line).
|
||||||
|
|
||||||
### Command-line Interface (`crunchyroll`)
|
### Command-line Interface (`crunchy`)
|
||||||
|
|
||||||
The [command-line interface](http://en.wikipedia.org/wiki/Command-line_interface) does not have a graphical component and is ideal for automation purposes and headless machines. The interface can run using a sequence of series addresses (the site address containing the episode listing), or with a batch-mode source file. The `crunchyroll --help` command will produce the following output:
|
The [command-line interface](http://en.wikipedia.org/wiki/Command-line_interface) does not have a graphical component and is ideal for automation purposes and headless machines. The interface can run using a sequence of series addresses (the site address containing the episode listing), or with a batch-mode source file. The `crunchy --help` command will produce the following output:
|
||||||
|
|
||||||
Usage: crunchyroll [options]
|
Usage: crunchy [options]
|
||||||
|
|
||||||
Options:
|
Options:
|
||||||
|
|
||||||
@@ -74,15 +76,15 @@ When no sequence of series addresses is provided, the batch-mode source file wil
|
|||||||
|
|
||||||
Download in batch-mode:
|
Download in batch-mode:
|
||||||
|
|
||||||
crunchyroll
|
crunchy
|
||||||
|
|
||||||
Download *Fairy Tail* to the current work directory:
|
Download *Fairy Tail* to the current work directory:
|
||||||
|
|
||||||
crunchyroll http://www.crunchyroll.com/fairy-tail
|
crunchy http://www.crunchyroll.com/fairy-tail
|
||||||
|
|
||||||
Download *Fairy Tail* to `C:\Anime`:
|
Download *Fairy Tail* to `C:\Anime`:
|
||||||
|
|
||||||
crunchyroll --output C:\Anime http://www.crunchyroll.com/fairy-tail
|
crunchy --output C:\Anime http://www.crunchyroll.com/fairy-tail
|
||||||
|
|
||||||
#### Switches
|
#### Switches
|
||||||
|
|
||||||
|
|||||||
15
package.json
15
package.json
@@ -1,19 +1,20 @@
|
|||||||
{
|
{
|
||||||
"author": "Roel van Uden",
|
"author": "Godzil",
|
||||||
"description": "CrunchyRoll.js is capable of downloading anime episodes from the popular CrunchyRoll streaming service.",
|
"description": "Crunchy.js is a fork of Crunchyroll.js, capable of downloading anime episodes from the popular CrunchyRoll streaming service.",
|
||||||
|
"license": "MIT",
|
||||||
"keywords": [
|
"keywords": [
|
||||||
"anime",
|
"anime",
|
||||||
"download",
|
"download",
|
||||||
"crunchyroll"
|
"crunchyroll"
|
||||||
],
|
],
|
||||||
"name": "crunchyroll",
|
"name": "crunchy",
|
||||||
"repository": {
|
"repository": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "git://github.com/Deathspike/crunchyroll.js.git"
|
"url": "git://github.com/Godzil/crunchyroll.js.git"
|
||||||
},
|
},
|
||||||
"version": "1.1.5",
|
"version": "1.1.10",
|
||||||
"bin": {
|
"bin": {
|
||||||
"crunchyroll": "./bin/crunchyroll"
|
"crunchy": "./bin/crunchy"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"big-integer": "1.4.4",
|
"big-integer": "1.4.4",
|
||||||
@@ -30,7 +31,7 @@
|
|||||||
},
|
},
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"prepublish": "npm run tsd && tsc",
|
"prepublish": "npm run tsd && tsc",
|
||||||
"test": "node ts --only-test",
|
"test": "node ts --only-test",
|
||||||
"tsd": "tsd reinstall -o -s"
|
"tsd": "tsd reinstall -o -s"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
107
src/episode.ts
107
src/episode.ts
@@ -11,11 +11,11 @@ import xml2js = require('xml2js');
|
|||||||
/**
|
/**
|
||||||
* Streams the episode to disk.
|
* Streams the episode to disk.
|
||||||
*/
|
*/
|
||||||
export default function(config: IConfig, address: string, done: (err: Error) => void) {
|
export default function(config: IConfig, address: string, done: (err: Error, ign: boolean) => void) {
|
||||||
scrapePage(config, address, (err, page) => {
|
scrapePage(config, address, (err, page) => {
|
||||||
if (err) return done(err);
|
if (err) return done(err, false);
|
||||||
scrapePlayer(config, address, page.id, (err, player) => {
|
scrapePlayer(config, address, page.id, (err, player) => {
|
||||||
if (err) return done(err);
|
if (err) return done(err, false);
|
||||||
download(config, page, player, done);
|
download(config, page, player, done);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
@@ -24,37 +24,72 @@ export default function(config: IConfig, address: string, done: (err: Error) =>
|
|||||||
/**
|
/**
|
||||||
* Completes a download and writes the message with an elapsed time.
|
* Completes a download and writes the message with an elapsed time.
|
||||||
*/
|
*/
|
||||||
function complete(message: string, begin: number, done: (err: Error) => void) {
|
function complete(message: string, begin: number, done: (err: Error, ign: boolean) => void) {
|
||||||
var timeInMs = Date.now() - begin;
|
var timeInMs = Date.now() - begin;
|
||||||
var seconds = prefix(Math.floor(timeInMs / 1000) % 60, 2);
|
var seconds = prefix(Math.floor(timeInMs / 1000) % 60, 2);
|
||||||
var minutes = prefix(Math.floor(timeInMs / 1000 / 60) % 60, 2);
|
var minutes = prefix(Math.floor(timeInMs / 1000 / 60) % 60, 2);
|
||||||
var hours = prefix(Math.floor(timeInMs / 1000 / 60 / 60), 2);
|
var hours = prefix(Math.floor(timeInMs / 1000 / 60 / 60), 2);
|
||||||
console.log(message + ' (' + hours + ':' + minutes + ':' + seconds + ')');
|
console.log(message + ' (' + hours + ':' + minutes + ':' + seconds + ')');
|
||||||
done(null);
|
done(null, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check if a file exist..
|
||||||
|
*/
|
||||||
|
function fileExist(path: string) {
|
||||||
|
try
|
||||||
|
{
|
||||||
|
fs.statSync(path);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
catch (e) { }
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Downloads the subtitle and video.
|
* Downloads the subtitle and video.
|
||||||
*/
|
*/
|
||||||
function download(config: IConfig, page: IEpisodePage, player: IEpisodePlayer, done: (err: Error) => void) {
|
function download(config: IConfig, page: IEpisodePage, player: IEpisodePlayer, done: (err: Error, ign: boolean) => void) {
|
||||||
var series = config.series || page.series;
|
var series = config.series || page.series;
|
||||||
var fileName = name(config, page, series);
|
series = series.replace("/","_").replace("'","_").replace(":","_");
|
||||||
|
var fileName = name(config, page, series, "").replace("/","_").replace("'","_").replace(":","_");
|
||||||
var filePath = path.join(config.output || process.cwd(), series, fileName);
|
var filePath = path.join(config.output || process.cwd(), series, fileName);
|
||||||
|
if (fileExist(filePath + ".mkv"))
|
||||||
|
{
|
||||||
|
var count = 0;
|
||||||
|
console.info("File '"+fileName+"' already exist...");
|
||||||
|
do
|
||||||
|
{
|
||||||
|
count = count + 1;
|
||||||
|
fileName = name(config, page, series, "-" + count).replace("/","_").replace("'","_").replace(":","_");
|
||||||
|
filePath = path.join(config.output || process.cwd(), series, fileName);
|
||||||
|
} while(fileExist(filePath + ".mkv"))
|
||||||
|
console.info("Renaming to '"+fileName+"'...");
|
||||||
|
}
|
||||||
|
|
||||||
mkdirp(path.dirname(filePath), (err: Error) => {
|
mkdirp(path.dirname(filePath), (err: Error) => {
|
||||||
if (err) return done(err);
|
if (err) return done(err, false);
|
||||||
downloadSubtitle(config, player, filePath, err => {
|
downloadSubtitle(config, player, filePath, err => {
|
||||||
if (err) return done(err);
|
if (err) return done(err, false);
|
||||||
var now = Date.now();
|
var now = Date.now();
|
||||||
console.log('Fetching ' + fileName);
|
if (player.video.file != undefined)
|
||||||
downloadVideo(config, page, player, filePath, err => {
|
{
|
||||||
if (err) return done(err);
|
console.log('Fetching ' + fileName);
|
||||||
if (config.merge) return complete('Finished ' + fileName, now, done);
|
downloadVideo(config, page, player, filePath, err => {
|
||||||
var isSubtited = Boolean(player.subtitle);
|
if (err) return done(err, false);
|
||||||
video.merge(config, isSubtited, player.video.file, filePath, player.mode, err => {
|
if (config.merge) return complete('Finished ' + fileName, now, done);
|
||||||
if (err) return done(err);
|
var isSubtited = Boolean(player.subtitle);
|
||||||
complete('Finished ' + fileName, now, done);
|
video.merge(config, isSubtited, player.video.file, filePath, player.video.mode, err => {
|
||||||
|
if (err) return done(err, false);
|
||||||
|
complete('Finished ' + fileName, now, done);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
});
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
console.log('Ignoring ' + fileName + ': not released yet');
|
||||||
|
done(null, true);
|
||||||
|
}
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@@ -96,11 +131,13 @@ function downloadVideo(config: IConfig,
|
|||||||
/**
|
/**
|
||||||
* Names the file based on the config, page, series and tag.
|
* Names the file based on the config, page, series and tag.
|
||||||
*/
|
*/
|
||||||
function name(config: IConfig, page: IEpisodePage, series: string) {
|
function name(config: IConfig, page: IEpisodePage, series: string, extra: string) {
|
||||||
var episode = (page.episode < 10 ? '0' : '') + page.episode;
|
var episodeNum = parseInt(page.episode, 10);
|
||||||
var volume = (page.volume < 10 ? '0' : '') + page.volume;
|
var volumeNum = parseInt(page.volume, 10);
|
||||||
|
var episode = (episodeNum < 10 ? '0' : '') + page.episode;
|
||||||
|
var volume = (volumeNum < 10 ? '0' : '') + page.volume;
|
||||||
var tag = config.tag || 'CrunchyRoll';
|
var tag = config.tag || 'CrunchyRoll';
|
||||||
return series + ' ' + volume + 'x' + episode + ' [' + tag + ']';
|
return series + ' - s' + volume + 'e' + episode +' - [' + tag + ']' + extra;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -122,15 +159,29 @@ function scrapePage(config: IConfig, address: string, done: (err: Error, page?:
|
|||||||
if (err) return done(err);
|
if (err) return done(err);
|
||||||
var $ = cheerio.load(result);
|
var $ = cheerio.load(result);
|
||||||
var swf = /^([^?]+)/.exec($('link[rel=video_src]').attr('href'));
|
var swf = /^([^?]+)/.exec($('link[rel=video_src]').attr('href'));
|
||||||
var regexp = /-\s+(?:Watch\s+)?(.+?)(?:\s+Season\s+([0-9]+))?(?:\s+-)?\s+Episode\s+([0-9]+)/;
|
var regexp = /\s*([^\n\r\t\f]+)\n?\s*[^0-9]*([0-9][0-9.]*)?,?\n?\s\s*[^0-9]*((PV )?[S0-9][P0-9.]*[a-fA-F]?)/;
|
||||||
var data = regexp.exec($('title').text());
|
var look = $('#showmedia_about_media').text();
|
||||||
if (!swf || !data) return done(new Error('Invalid page.'));
|
var seasonTitle = $('span[itemprop="title"]').text();
|
||||||
|
var data = regexp.exec(look);
|
||||||
|
|
||||||
|
if (!swf || !data)
|
||||||
|
{
|
||||||
|
console.info('Something wrong in the page at '+address+' (data are: '+look+')');
|
||||||
|
console.info('Setting Season to 0 and episode to \’0\’...');
|
||||||
|
done(null, {
|
||||||
|
id: id,
|
||||||
|
episode: "0",
|
||||||
|
series: seasonTitle,
|
||||||
|
swf: swf[1],
|
||||||
|
volume: "0"
|
||||||
|
});
|
||||||
|
}
|
||||||
done(null, {
|
done(null, {
|
||||||
id: id,
|
id: id,
|
||||||
episode: parseInt(data[3], 10),
|
episode: data[3],
|
||||||
series: data[1],
|
series: data[1],
|
||||||
swf: swf[1],
|
swf: swf[1],
|
||||||
volume: parseInt(data[2], 10) || 1
|
volume: data[2] || "1"
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@@ -165,7 +216,7 @@ function scrapePlayer(config: IConfig, address: string, id: number, done: (err:
|
|||||||
data: player['default:preload'].subtitle.data
|
data: player['default:preload'].subtitle.data
|
||||||
} : null,
|
} : null,
|
||||||
video: {
|
video: {
|
||||||
mode: streamMode;
|
mode: streamMode,
|
||||||
file: player['default:preload'].stream_info.file,
|
file: player['default:preload'].stream_info.file,
|
||||||
host: player['default:preload'].stream_info.host
|
host: player['default:preload'].stream_info.host
|
||||||
}
|
}
|
||||||
|
|||||||
4
src/interface/IEpisodePage.d.ts
vendored
4
src/interface/IEpisodePage.d.ts
vendored
@@ -1,7 +1,7 @@
|
|||||||
interface IEpisodePage {
|
interface IEpisodePage {
|
||||||
id: number;
|
id: number;
|
||||||
episode: number;
|
episode: string;
|
||||||
series: string;
|
series: string;
|
||||||
volume: number;
|
volume: string;
|
||||||
swf: string;
|
swf: string;
|
||||||
}
|
}
|
||||||
|
|||||||
1
src/interface/IEpisodePlayer.d.ts
vendored
1
src/interface/IEpisodePlayer.d.ts
vendored
@@ -5,6 +5,7 @@ interface IEpisodePlayer {
|
|||||||
data: string;
|
data: string;
|
||||||
};
|
};
|
||||||
video: {
|
video: {
|
||||||
|
mode: string;
|
||||||
file: string;
|
file: string;
|
||||||
host: string;
|
host: string;
|
||||||
};
|
};
|
||||||
|
|||||||
2
src/interface/ISeriesEpisode.d.ts
vendored
2
src/interface/ISeriesEpisode.d.ts
vendored
@@ -1,5 +1,5 @@
|
|||||||
interface ISeriesEpisode {
|
interface ISeriesEpisode {
|
||||||
address: string;
|
address: string;
|
||||||
episode: number;
|
episode: string;
|
||||||
volume: number;
|
volume: number;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -19,14 +19,22 @@ export default function(config: IConfig, address: string, done: (err: Error) =>
|
|||||||
var i = 0;
|
var i = 0;
|
||||||
(function next() {
|
(function next() {
|
||||||
if (i >= page.episodes.length) return done(null);
|
if (i >= page.episodes.length) return done(null);
|
||||||
download(cache, config, address, page.episodes[i], err => {
|
download(cache, config, address, page.episodes[i], (err, ignored) => {
|
||||||
if (err) return done(err);
|
if (err) return done(err);
|
||||||
var newCache = JSON.stringify(cache, null, ' ');
|
if ((ignored == false) || (ignored == undefined))
|
||||||
fs.writeFile(persistentPath, newCache, err => {
|
{
|
||||||
if (err) return done(err);
|
var newCache = JSON.stringify(cache, null, ' ');
|
||||||
|
fs.writeFile(persistentPath, newCache, err => {
|
||||||
|
if (err) return done(err);
|
||||||
|
i += 1;
|
||||||
|
next();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
i += 1;
|
i += 1;
|
||||||
next();
|
next();
|
||||||
});
|
}
|
||||||
});
|
});
|
||||||
})();
|
})();
|
||||||
});
|
});
|
||||||
@@ -40,14 +48,14 @@ function download(cache: {[address: string]: number},
|
|||||||
config: IConfig,
|
config: IConfig,
|
||||||
baseAddress: string,
|
baseAddress: string,
|
||||||
item: ISeriesEpisode,
|
item: ISeriesEpisode,
|
||||||
done: (err: Error) => void) {
|
done: (err: Error, ign: boolean) => void) {
|
||||||
if (!filter(config, item)) return done(null);
|
if (!filter(config, item)) return done(null, false);
|
||||||
var address = url.resolve(baseAddress, item.address);
|
var address = url.resolve(baseAddress, item.address);
|
||||||
if (cache[address]) return done(null);
|
if (cache[address]) return done(null, false);
|
||||||
episode(config, address, err => {
|
episode(config, address, (err, ignored) => {
|
||||||
if (err) return done(err);
|
if (err) return done(err, false);
|
||||||
cache[address] = Date.now();
|
cache[address] = Date.now();
|
||||||
done(null);
|
done(null, ignored);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -57,8 +65,8 @@ function download(cache: {[address: string]: number},
|
|||||||
function filter(config: IConfig, item: ISeriesEpisode) {
|
function filter(config: IConfig, item: ISeriesEpisode) {
|
||||||
// Filter on chapter.
|
// Filter on chapter.
|
||||||
var episodeFilter = config.episode;
|
var episodeFilter = config.episode;
|
||||||
if (episodeFilter > 0 && item.episode <= episodeFilter) return false;
|
if (episodeFilter > 0 && parseInt(item.episode, 10) <= episodeFilter) return false;
|
||||||
if (episodeFilter < 0 && item.episode >= -episodeFilter) return false;
|
if (episodeFilter < 0 && parseInt(item.episode, 10) >= -episodeFilter) return false;
|
||||||
|
|
||||||
// Filter on volume.
|
// Filter on volume.
|
||||||
var volumeFilter = config.volume;
|
var volumeFilter = config.volume;
|
||||||
@@ -75,18 +83,18 @@ function page(config: IConfig, address: string, done: (err: Error, result?: ISer
|
|||||||
if (err) return done(err);
|
if (err) return done(err);
|
||||||
var $ = cheerio.load(result);
|
var $ = cheerio.load(result);
|
||||||
var title = $('span[itemprop=name]').text();
|
var title = $('span[itemprop=name]').text();
|
||||||
if (!title) return done(new Error('Invalid page.'));
|
if (!title) return done(new Error('Invalid page.(' + address + ')'));
|
||||||
var episodes: ISeriesEpisode[] = [];
|
var episodes: ISeriesEpisode[] = [];
|
||||||
$('.episode').each((i, el) => {
|
$('.episode').each((i, el) => {
|
||||||
if ($(el).children('img[src*=coming_soon]').length) return;
|
if ($(el).children('img[src*=coming_soon]').length) return;
|
||||||
var volume = /([0-9]+)\s*$/.exec($(el).closest('ul').prev('a').text());
|
var volume = /([0-9]+)\s*$/.exec($(el).closest('ul').prev('a').text());
|
||||||
var regexp = /Episode\s+([0-9]+)\s*$/i;
|
var regexp = /Episode\s+((PV )?[S0-9][P0-9.]*[a-fA-F]?)\s*$/i;
|
||||||
var episode = regexp.exec($(el).children('.series-title').text());
|
var episode = regexp.exec($(el).children('.series-title').text());
|
||||||
var address = $(el).attr('href');
|
var address = $(el).attr('href');
|
||||||
if (!address || !episode) return;
|
if (!address || !episode) return;
|
||||||
episodes.push({
|
episodes.push({
|
||||||
address: address,
|
address: address,
|
||||||
episode: parseInt(episode[0], 10),
|
episode: episode[1],
|
||||||
volume: volume ? parseInt(volume[0], 10) : 1
|
volume: volume ? parseInt(volume[0], 10) : 1
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
Reference in New Issue
Block a user