551 lines
16 KiB
JavaScript
551 lines
16 KiB
JavaScript
const fs = require('node:fs/promises');
|
|
const path = require('path');
|
|
const HTMLparser = require('node-html-parser');
|
|
const exec = require("child_process").execSync;
|
|
const nodemailer = require('nodemailer');
|
|
|
|
var transporter;
|
|
var config;
|
|
|
|
function clone(obj) {
|
|
return JSON.parse(JSON.stringify(obj));
|
|
}
|
|
|
|
function cleanPath(pathToClean) {
|
|
let regex = /[<>:"|\?\*]/g;
|
|
let cleanPath = path.normalize(pathToClean);
|
|
|
|
let isAbsolute = path.isAbsolute(cleanPath);
|
|
cleanPath = cleanPath.replace(regex, '');
|
|
if (/^win/i.test(process.platform) && isAbsolute) {
|
|
cleanPath = `${cleanPath.slice(0, 1)}:${cleanPath.slice(1)}`;
|
|
}
|
|
|
|
return cleanPath;
|
|
}
|
|
|
|
function padNumber(num, len) {
|
|
let strNum = num.toString();
|
|
let negative = num < 0;
|
|
if (negative) strNum = strNum.substr(1);
|
|
|
|
while (strNum.length < len) {
|
|
strNum = '0' + strNum;
|
|
}
|
|
|
|
return (negative ? '-' : '') + strNum;
|
|
}
|
|
|
|
function log(text) {
|
|
let d = new Date();
|
|
let dateTime = `${d.getFullYear()}.${padNumber((d.getMonth() + 1), 2)}.${padNumber(d.getDate(), 2)}_${padNumber(d.getHours(), 2)}:${padNumber(d.getMinutes(), 2)}:${padNumber(d.getSeconds(), 2)}.${padNumber(d.getMilliseconds(), 3)}`;
|
|
fs.appendFile(cleanPath(`${__dirname}/WNtoEmail.log`), `${dateTime} ${text}\n`);
|
|
fs.appendFile(cleanPath(`${__dirname}/WNtoEmailArch.log`), `${dateTime} ${text}\n`);
|
|
console.log(text);
|
|
}
|
|
|
|
async function mkDir(dirPath) {
|
|
try {
|
|
await fs.access(cleanPath(dirPath));
|
|
}
|
|
catch (err) {
|
|
await fs.mkdir(cleanPath(dirPath), { recursive: true });
|
|
}
|
|
await fs.access(cleanPath(dirPath));
|
|
}
|
|
|
|
async function writeFile(dir, file, data) {
|
|
let cleanDir = cleanPath(dir);
|
|
let cleanFile = cleanPath(file);
|
|
|
|
await mkDir(dir);
|
|
|
|
return fs.writeFile(`${cleanDir}/${cleanFile}`, data);
|
|
}
|
|
|
|
async function readFile(file, { format = 'utf8' } = {}) {
|
|
return fs.readFile(cleanPath(file), format);
|
|
}
|
|
|
|
async function loadConfig() {
|
|
let novelConfigDefault = {
|
|
"downloadLocation": "",
|
|
"converterPath": "D:/Calibre/Calibre/ebook-convert.exe",
|
|
"ebookFormat": "epub",
|
|
"sendEmail": false,
|
|
"copyPath": "",
|
|
"emailToAddress": "",
|
|
"emailFromAddress": "",
|
|
"emailProvider": "",
|
|
"emailUsername": "",
|
|
"emailPassword": "",
|
|
"emailAttachments": 25,
|
|
"supportedHosting": {
|
|
"NF": "https://novelfull.com/",
|
|
"TNC": "https://thatnovelcorner.com/ external source, use with sendOnly = true",
|
|
"BBB": "https://bluebellsinbloom.wordpress.com/",
|
|
"JN": "https://jnovels.com/ external"
|
|
},
|
|
"template": {
|
|
"novelURL": "",
|
|
"title": "",
|
|
"author": "",
|
|
"coverURL": "",
|
|
"lastChapterURL": "",
|
|
"lastVolume": 0,
|
|
"volumePadding": 2,
|
|
"completed": false,
|
|
"hosting": "NF",
|
|
"volumeChapterCount": 5,
|
|
"completedVolumeChapterCount": 200,
|
|
"redownload": false,
|
|
"sendOnly": false,
|
|
"sendOnlyFormat": "epub",
|
|
"sendOnlyConvert": true,
|
|
"sendOnlyRegex": "(?<volume>\\d*). (?<title>.*); (?<author>.*)"
|
|
},
|
|
"novels": []
|
|
};
|
|
|
|
try {
|
|
config = JSON.parse(await readFile(`${__dirname}/novelConfig.json`));
|
|
|
|
for (key in novelConfigDefault) {
|
|
if (config[key] == undefined) {
|
|
config[key] = clone(novelConfigDefault[key]);
|
|
}
|
|
}
|
|
|
|
for (key in novelConfigDefault['template']) {
|
|
if (config['template'][key] == undefined) {
|
|
config['template'][key] = clone(novelConfigDefault['template'][key]);
|
|
}
|
|
}
|
|
|
|
for (let i = 0; i < config['novels'].length; ++i) {
|
|
for (key in novelConfigDefault['template']) {
|
|
if (config['novels'][i][key] == undefined) {
|
|
config['novels'][i][key] = clone(novelConfigDefault['template'][key]);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
catch (err) {
|
|
config = clone(novelConfigDefault);
|
|
}
|
|
|
|
await saveConfig();
|
|
await saveConfig({ configName: 'novelConfig.bak.json' });
|
|
|
|
transporter = nodemailer.createTransport({
|
|
service: config['emailProvider'],
|
|
auth: {
|
|
user: config['emailUsername'],
|
|
pass: config['emailPassword']
|
|
}
|
|
});
|
|
}
|
|
|
|
async function saveConfig({ configPath = __dirname, configName = 'novelConfig.json' } = {}) {
|
|
await writeFile(configPath, configName, JSON.stringify(config, null, 4));
|
|
}
|
|
|
|
async function convertEbook(dir, file, { cover = false, authors = false, title = false, format = 'html', file2 = false } = {}) {
|
|
let file1Path = cleanPath(`${dir}/${file}.${format}`);
|
|
let file2Path = cleanPath(`${dir}/${file2 ? file2 : file}.${config['ebookFormat']}`);
|
|
let convertParams = ' --use-auto-toc';
|
|
convertParams += config['ebookFormat'] == 'epub' ? ' --epub-inline-toc' : '';
|
|
convertParams += cover ? ` --cover "${cover}"` : '';
|
|
convertParams += authors ? ` --authors "${authors}"` : '';
|
|
convertParams += title ? ` --title "${title}"` : '';
|
|
|
|
log(`Converting volume: ${file1Path}`);
|
|
|
|
let convertOutput = exec(`${config['converterPath']} "${file1Path}" "${file2Path}"${convertParams}`, function (error, stdout, stderr) {
|
|
if (error) {
|
|
log(`error: ${error.message}`);
|
|
return;
|
|
}
|
|
if (stderr) {
|
|
log(`stderr: ${stderr}`);
|
|
return;
|
|
}
|
|
log(`stdout: ${stdout}`);
|
|
});
|
|
|
|
log(convertOutput.toString());
|
|
}
|
|
|
|
async function sendEbook(subject, ebookAttachments) {
|
|
if (config['sendEmail']) {
|
|
let splicedAttachments = [];
|
|
while (ebookAttachments.length > config['emailAttachments']) {
|
|
splicedAttachments.push(ebookAttachments.splice(0, config['emailAttachments']));
|
|
}
|
|
if (ebookAttachments.length > 0) {
|
|
splicedAttachments.push(ebookAttachments);
|
|
}
|
|
|
|
for (let i = 0; i < splicedAttachments.length; ++i) {
|
|
let message = {
|
|
from: config['emailFromAddress'],
|
|
to: config['emailToAddress'],
|
|
subject: subject + ' part ' + (i + 1),
|
|
text: subject + ' part ' + (i + 1),
|
|
attachments: splicedAttachments[i]
|
|
};
|
|
|
|
transporter.sendMail(message, (err) => {
|
|
if (err)
|
|
log(`Send mail error: ${err}`);
|
|
});
|
|
|
|
let sentVolumes = '';
|
|
splicedAttachments[i].forEach(elem => { sentVolumes += '\n' + elem['filename']; });
|
|
log(`Sent volumes:${sentVolumes}`);
|
|
}
|
|
}
|
|
if (config['copyPath'] != "") {
|
|
for (const ebook of ebookAttachments) {
|
|
await mkDir(`${config.copyPath}\\${ebook.title}`);
|
|
fs.copyFile(ebook.path, cleanPath(`${config.copyPath}\\${ebook.title}\\${ebook.filename}`),);
|
|
}
|
|
}
|
|
}
|
|
|
|
async function fetch_something(URL, hosting) {
|
|
let fetchURL = await fetch(URL);
|
|
|
|
if (fetchURL.ok) {
|
|
let response = await fetchURL.text();
|
|
|
|
let info;
|
|
|
|
return info;
|
|
}
|
|
else return fetchURL.ok;
|
|
}
|
|
|
|
async function fetchNovelInfo(URL, hosting) {
|
|
let fetchURL = await fetch(URL);
|
|
|
|
if (fetchURL.ok) {
|
|
let response = await fetchURL.text();
|
|
|
|
let novelInfo = getNovelInfo(response, hosting);
|
|
|
|
return novelInfo;
|
|
}
|
|
else return fetchURL.ok;
|
|
|
|
}
|
|
|
|
async function fetchChapter(URL, hosting) {
|
|
let fetchURL = await fetch(URL);
|
|
|
|
if (fetchURL.ok) {
|
|
let response = await fetchURL.text();
|
|
|
|
let nextChapterURL = getNextChapterURL(response, hosting);
|
|
let chapterContent = getChapterContent(response, hosting);
|
|
|
|
return [chapterContent, nextChapterURL];
|
|
}
|
|
else return fetchURL.ok;
|
|
}
|
|
|
|
function get(response, hosting) {
|
|
let html = HTMLparser.parse(response);
|
|
|
|
let info;
|
|
|
|
switch (hosting) {
|
|
case 'NF':
|
|
info = html;
|
|
break;
|
|
|
|
default:
|
|
info = false;
|
|
}
|
|
|
|
return info;
|
|
}
|
|
|
|
async function getNovelInfo(response, hosting) {
|
|
let html = HTMLparser.parse(response);
|
|
|
|
let info;
|
|
|
|
switch (hosting) {
|
|
case 'NF':
|
|
let title = html.querySelector('h3.title').innerText;
|
|
let author = html.querySelector('#truyen > div.csstransforms3d > div > div.col-xs-12.col-info-desc > div.col-xs-12.col-sm-4.col-md-4.info-holder > div.info > div:nth-child(1) > a:nth-child(2)').innerText;
|
|
let completed = html.querySelector('#truyen > div.csstransforms3d > div > div.col-xs-12.col-info-desc > div.col-xs-12.col-sm-4.col-md-4.info-holder > div.info > div:nth-child(5) > a').innerText == "Completed";
|
|
let firstChapterURL = html.querySelector('#list-chapter > div.row > div:nth-child(1) > ul > li:nth-child(1) > a').attrs['href'];
|
|
let coverURL = html.querySelector('div.book > img').attrs['src'];
|
|
info = [title, author, completed, 'https://novelfull.com' + firstChapterURL, 'https://novelfull.com' + coverURL];
|
|
break;
|
|
|
|
case 'JN':
|
|
html.querySelectorAll('.post-content ol li').forEach(elem => {
|
|
if (elem.innerText.match('V|volume')) {
|
|
info = elem.innerText.match(/\d+/)[0];
|
|
}
|
|
});
|
|
break;
|
|
|
|
case 'TNC':
|
|
html.querySelectorAll('a').forEach(elem => {
|
|
if (elem.innerText.match('V|volume')) {
|
|
info = elem.innerText.match(/\d+/)[0];
|
|
}
|
|
});
|
|
|
|
break;
|
|
|
|
default:
|
|
info = false;
|
|
}
|
|
|
|
return info;
|
|
}
|
|
|
|
function getNextChapterURL(response, hosting) {
|
|
let html = HTMLparser.parse(response);
|
|
|
|
let nextChapterURL;
|
|
|
|
switch (hosting) {
|
|
case 'NF':
|
|
nextChapterURL = undefined == html.querySelector('a#next_chap').attrs['href'] ? false : 'https://novelfull.com' + html.querySelector('a#next_chap').attrs['href'];
|
|
break;
|
|
|
|
default:
|
|
nextChapterURL = false;
|
|
}
|
|
|
|
return nextChapterURL;
|
|
}
|
|
|
|
function getChapterContent(response, hosting) {
|
|
let html = HTMLparser.parse(response);
|
|
|
|
let chapterContent = '';
|
|
|
|
switch (hosting) {
|
|
case 'NF':
|
|
chapterContent += '<h1 class="chapter">' + html.querySelector('span.chapter-text').innerText + '</h1>';
|
|
|
|
html.querySelectorAll('div#chapter-content p').forEach(element => {
|
|
chapterContent += element.outerHTML;
|
|
});
|
|
break;
|
|
|
|
default:
|
|
chapterContent = false;
|
|
}
|
|
|
|
return chapterContent;
|
|
}
|
|
|
|
async function clearLog() {
|
|
await writeFile(`${__dirname}`, `WNtoEmail.log`, '');
|
|
}
|
|
|
|
async function main() {
|
|
await loadConfig();
|
|
await clearLog();
|
|
|
|
for (let i = 0; i < config['novels'].length; ++i) {
|
|
let novel = clone(config['novels'][i]);
|
|
|
|
if (novel['redownload']) {
|
|
novel['completed'] = false;
|
|
novel['lastChapterURL'] = false;
|
|
novel['lastVolume'] = 0;
|
|
}
|
|
|
|
if (!novel['completed']) {
|
|
if (novel.sendOnly) {
|
|
await sendOnlyFunction(novel, i);
|
|
} else {
|
|
await downloadChaptersFunction(novel, i);
|
|
}
|
|
}
|
|
}
|
|
|
|
async function sendOnlyFunction(novel, i) {
|
|
let ebookAttachments = [];
|
|
const novelPath = `${config.downloadLocation}/${novel.title}`;
|
|
const novelVolumeRegex = new RegExp(novel.sendOnlyRegex + '.' + novel.sendOnlyFormat);
|
|
|
|
let lastVolumeOnline = parseInt(await fetchNovelInfo(novel['novelURL'], novel.hosting));
|
|
|
|
if (lastVolumeOnline > novel.lastVolume) {
|
|
log(`New volume found online: ${novel.title} ${lastVolumeOnline} ${novel.novelURL}`);
|
|
}
|
|
|
|
// let ebookList = await fs.opendir(cleanPath(novelPath));
|
|
// console.log(ebookList);
|
|
|
|
try {
|
|
const files = await fs.readdir(cleanPath(novelPath));
|
|
for (const file of files) {
|
|
let volumeMatch = file.match(novelVolumeRegex);
|
|
if (volumeMatch) {
|
|
const currentVolume = parseInt(volumeMatch.groups.volume);
|
|
if (currentVolume > novel.lastVolume) {
|
|
if (novel.sendOnlyConvert) {
|
|
convertEbook(novelPath, path.parse(file).name, { format: novel.sendOnlyFormat, title: `${padNumber(currentVolume, novel.volumePadding)}. ${novel.title}`, file2: `${padNumber(currentVolume, novel.volumePadding)}. ${novel.title}` });
|
|
|
|
ebookAttachments.push({
|
|
title: novel.title,
|
|
filename: cleanPath(`${padNumber(currentVolume, novel.volumePadding)}. ${novel.title}.${config.ebookFormat}`),
|
|
path: cleanPath(`${novelPath}/${padNumber(currentVolume, novel.volumePadding)}. ${novel.title}.${config.ebookFormat}`)
|
|
});
|
|
|
|
} else {
|
|
ebookAttachments.push({
|
|
title: novel.title,
|
|
filename: cleanPath(`${file}`),
|
|
path: cleanPath(`${novelPath}/${file}`)
|
|
});
|
|
}
|
|
novel.lastVolume = currentVolume;
|
|
}
|
|
}
|
|
}
|
|
} catch (err) {
|
|
log(err);
|
|
}
|
|
|
|
config['novels'][i] = clone(novel);
|
|
await saveConfig();
|
|
|
|
sendEbook(novel['title'], ebookAttachments);
|
|
}
|
|
|
|
async function downloadChaptersFunction(novel, i) {
|
|
let chapters = [];
|
|
let nextChapterURL;
|
|
let novelInfo = await fetchNovelInfo(novel['novelURL'], novel.hosting);
|
|
|
|
novel['title'] = novelInfo[0];
|
|
novel['author'] = novelInfo[1];
|
|
novel['completed'] = novelInfo[2];
|
|
novel['coverURL'] = novelInfo[4];
|
|
|
|
config['novels'][i] = clone(novel);
|
|
await saveConfig();
|
|
|
|
if (!novel['lastChapterURL']) {
|
|
novel['lastChapterURL'] = novelInfo[3];
|
|
|
|
let chapter = await fetchChapter(novelInfo[3], novel.hosting);
|
|
log('Downloaded chapter: ' + chapters.length + ' ' + novelInfo[3]);
|
|
chapters.push(chapter);
|
|
}
|
|
|
|
let novelDir = `${config['downloadLocation']}/${novel['title']}`;
|
|
|
|
const nextChapterURLtemp = await fetchChapter(novel['lastChapterURL'], novel.hosting);
|
|
nextChapterURL = nextChapterURLtemp[1];
|
|
|
|
while (nextChapterURL) {
|
|
novel['lastChapterURL'] = nextChapterURL;
|
|
let chapter = await fetchChapter(nextChapterURL, novel.hosting);
|
|
log('Downloaded chapter: ' + chapters.length + ' ' + nextChapterURL);
|
|
chapters.push(chapter);
|
|
nextChapterURL = chapter[1];
|
|
}
|
|
|
|
let startVol = novel['lastVolume'];
|
|
let totalChapters = chapters.length;
|
|
|
|
const maxVolumeComplete = novel['completed'] ? startVol + Math.ceil(totalChapters / novel['completedVolumeChapterCount']) : startVol + Math.floor(totalChapters / novel['completedVolumeChapterCount']);
|
|
const maxVolumeUpdate = novel['completed'] ? 0 : 1;// ? maxVolumeComplete : maxVolumeComplete + Math.floor((totalChapters - ((maxVolumeComplete - startVol) * novel.completedVolumeChapterCount)) / novel['volumeChapterCount'])
|
|
const maxVolLen = novel['volumePadding'] ? novel['volumePadding'] : (maxVolumeUpdate.toString().length < 2 ? 2 : maxVolumeUpdate.toString().length);
|
|
|
|
let ebookAttachments = [];
|
|
|
|
for (let vol = startVol; vol < maxVolumeComplete; vol++) {
|
|
let volContent = '';
|
|
|
|
let chap;
|
|
for (chap = 0; chap < novel['completedVolumeChapterCount'] && chap < chapters.length; chap++) {
|
|
volContent += chapters[chap][0] + '\n';
|
|
}
|
|
|
|
novel['lastChapterURL'] = chapters[(chap - 2 < 0 ? 0 : chap - 2)][1];
|
|
novel['lastVolume'] = vol + 1;
|
|
config['novels'][i] = clone(novel);
|
|
await saveConfig();
|
|
|
|
let novelFileName = `${padNumber((vol + 1), novel.volumePadding)}. ${novel['title']}; ${novel['author']}`;
|
|
|
|
await writeFile(novelDir, `${novelFileName}.html`, volContent);
|
|
log(`Saved volume: ${novelFileName}`);
|
|
|
|
await convertEbook(novelDir, novelFileName, {
|
|
cover: novel['coverURL'],
|
|
authors: novel['author'],
|
|
title: `${padNumber((vol + 1), novel.volumePadding)}. ${novel['title']}`
|
|
});
|
|
|
|
ebookAttachments.push({
|
|
title: novel.title,
|
|
filename: cleanPath(`${novelFileName}.${config['ebookFormat']}`),
|
|
path: cleanPath(`${novelDir}/${novelFileName}.${config['ebookFormat']}`)
|
|
});
|
|
|
|
chapters.splice(0, chap);
|
|
}
|
|
|
|
startVol = novel['lastVolume'];
|
|
totalChapters = chapters.length;
|
|
|
|
if (maxVolumeUpdate && totalChapters >= novel.volumeChapterCount) {
|
|
let vol = maxVolumeComplete;
|
|
let volContent = '';
|
|
|
|
let chap;
|
|
for (chap = 0; chap < Math.floor(totalChapters / novel.volumeChapterCount) * novel.volumeChapterCount; chap++) {
|
|
volContent += chapters[chap][0] + '\n';
|
|
}
|
|
|
|
novel['lastChapterURL'] = chapters[(chap - 2 < 0 ? 0 : chap - 2)][1];
|
|
novel['lastVolume'] = vol + 1;
|
|
config['novels'][i] = clone(novel);
|
|
await saveConfig();
|
|
|
|
let novelFileName = `${padNumber((vol + 1), novel.volumePadding)}. ${novel['title']}; ${novel['author']}`;
|
|
|
|
await writeFile(novelDir, `${novelFileName}.html`, volContent);
|
|
log(`Saved volume: ${novelFileName}`);
|
|
|
|
await convertEbook(novelDir, novelFileName, {
|
|
cover: novel['coverURL'],
|
|
authors: novel['author'],
|
|
title: `${padNumber((vol + 1), novel.volumePadding)}. ${novel['title']}`
|
|
});
|
|
|
|
ebookAttachments.push({
|
|
title: novel.title,
|
|
filename: cleanPath(`${novelFileName}.${config['ebookFormat']}`),
|
|
path: cleanPath(`${novelDir}/${novelFileName}.${config['ebookFormat']}`)
|
|
});
|
|
|
|
chapters.splice(0, chap);
|
|
}
|
|
if (!novel['redownload']) {
|
|
sendEbook(novel['title'], ebookAttachments);
|
|
}
|
|
else {
|
|
novel['redownload'] = false;
|
|
config['novels'][i] = clone(novel);
|
|
saveConfig();
|
|
}
|
|
}
|
|
}
|
|
|
|
main(); |