Compare commits
144 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
428b507257 | ||
|
|
043dab0731 | ||
|
|
a7b4d9c0d8 | ||
|
|
6f9c95e351 | ||
|
|
7a53063ea8 | ||
|
|
ec4d5cac4f | ||
|
|
f8557cba88 | ||
|
|
5dead039f5 | ||
|
|
ea38392df4 | ||
|
|
0cc9d90a94 | ||
|
|
8c7b86c458 | ||
|
|
0e29546fc5 | ||
|
|
c9fa90d07c | ||
|
|
7d8e0525b1 | ||
|
|
ddf69876a6 | ||
|
|
1d78e75e38 | ||
|
|
7ed58fe3c6 | ||
|
|
058c79570b | ||
|
|
ec8fbcdf38 | ||
|
|
76673295bf | ||
|
|
084401b9c3 | ||
|
|
49038b10f7 | ||
|
|
45ea26810a | ||
|
|
18c8b2d803 | ||
|
|
f4a7482b3b | ||
|
|
32dff128f4 | ||
|
|
a00b2d6574 | ||
|
|
10c6e7d522 | ||
|
|
df6a256d51 | ||
|
|
fbdb74ee68 | ||
|
|
9ad7250da0 | ||
|
|
8c86984ea1 | ||
|
|
834b3f6210 | ||
|
|
105b8d5042 | ||
|
|
7ca8fd9ca1 | ||
|
|
0067c2800a | ||
|
|
688c8796f4 | ||
|
|
56af65742b | ||
|
|
629ad26d40 | ||
|
|
4b0e499c10 | ||
|
|
4697b46cba | ||
|
|
7f17e7daed | ||
|
|
a1fcb7597b | ||
|
|
35e46d0685 | ||
|
|
e2c0f3658b | ||
|
|
a3541ec16a | ||
|
|
08d0d3e7f3 | ||
|
|
2c47b2bee3 | ||
|
|
e6008b5ec4 | ||
|
|
e214ddf8d5 | ||
|
|
52927c6188 | ||
|
|
92ca9dd983 | ||
|
|
ed8be34c12 | ||
|
|
93bddfd05e | ||
|
|
8c99101bb3 | ||
|
|
d874f9ded4 | ||
|
|
d7be4d3d94 | ||
|
|
a2fa312839 | ||
|
|
f7e1e09928 | ||
|
|
f0832b07cb | ||
|
|
7c253df291 | ||
|
|
bb7cd9cbde | ||
|
|
56c4182985 | ||
|
|
cb6c7536bf | ||
|
|
fbfe8cbda0 | ||
|
|
6129d2d7eb | ||
|
|
16b30c922a | ||
|
|
c42ad66be6 | ||
|
|
f36c13fea1 | ||
|
|
4fd9d579e0 | ||
|
|
e65a8a13ea | ||
|
|
6ddb97d43e | ||
|
|
89082603de | ||
|
|
a9a3227433 | ||
|
|
60cb3514b2 | ||
|
|
4aeaa05f0b | ||
|
|
9c06552278 | ||
|
|
000f8dde82 | ||
|
|
9ffc218002 | ||
|
|
68a188f099 | ||
|
|
8829bb3810 | ||
|
|
5164d2f536 | ||
|
|
451538fcf7 | ||
|
|
82a02ef339 | ||
|
|
b834d4951f | ||
|
|
edc3b669be | ||
|
|
522826311d | ||
|
|
e69b9951d5 | ||
|
|
c6300222ea | ||
|
|
5aa6ee899c | ||
|
|
4b76f97d2b | ||
|
|
5ccfe71c55 | ||
|
|
97fc902cdb | ||
|
|
7e935951d7 | ||
|
|
810c6d68d2 | ||
|
|
003dc70f4f | ||
|
|
371ff64a95 | ||
|
|
b0de5adbf3 | ||
|
|
d1d2b07c33 | ||
|
|
d9b2444c1a | ||
|
|
e7fae27031 | ||
|
|
eb0c7b0a32 | ||
|
|
3d7ad0dd9a | ||
|
|
ae04feb311 | ||
|
|
7b59f911ef | ||
|
|
d3444da647 | ||
|
|
66738d0c9c | ||
|
|
7e187acd68 | ||
|
|
c751372a54 | ||
|
|
7fc98fc7da | ||
|
|
b56f45694e | ||
|
|
091ca521ef | ||
|
|
c7a17b0a76 | ||
|
|
26468b996a | ||
|
|
c4e240d87c | ||
|
|
04713f47c8 | ||
|
|
37ab3493db | ||
|
|
a4cb3c628e | ||
|
|
8492da8a13 | ||
|
|
98d7c64a56 | ||
|
|
25f121e5ed | ||
|
|
4c8797c99c | ||
|
|
1155aa285d | ||
|
|
239bbb8263 | ||
|
|
e6b9330108 | ||
|
|
935b767c2e | ||
|
|
8acf3295b5 | ||
|
|
48c3a12fa0 | ||
|
|
a1dea514b7 | ||
|
|
d4788439cb | ||
|
|
0a60ad354c | ||
|
|
c565a20344 | ||
|
|
735ee88f0b | ||
|
|
9405ce2cc0 | ||
|
|
115277d88a | ||
|
|
6925c11dbd | ||
|
|
984d835892 | ||
|
|
23353a4960 | ||
|
|
955bcda032 | ||
|
|
81ad5d7a2c | ||
|
|
dada7980ec | ||
|
|
511a308646 | ||
|
|
65c8f2cc81 | ||
|
|
238c18bc48 |
86
README.md
86
README.md
@@ -1,43 +1,43 @@
|
|||||||
# Liberama
|
# Liberama
|
||||||
|
|
||||||
Браузерная онлайн-читалка книг и децентрализованная библиотека.
|
Браузерная онлайн-читалка книг и децентрализованная библиотека.
|
||||||
|
|
||||||
Читалка <img src="https://omnireader.ru/favicon.ico" width="14px"/>[OmniReader](https://omnireader.ru) является частью данного проекта, размещенной на VPS:
|
Читалка <img src="https://omnireader.ru/favicon.ico" width="14px"/>[OmniReader](https://omnireader.ru) является частью данного проекта, размещенной на VPS:
|
||||||
|
|
||||||

|

|
||||||

|

|
||||||
|
|
||||||
## VPS
|
## VPS
|
||||||
Для разворачивания читалки на чистом VPS с нуля смотрите [docs/omnireader.ru](docs/omnireader.ru/README.md)
|
Для разворачивания читалки на чистом VPS с нуля смотрите [docs/omnireader.ru](docs/omnireader.ru/README.md)
|
||||||
|
|
||||||
## Сборка проекта
|
## Сборка проекта
|
||||||
Необходима версия node.js не ниже 14.
|
Необходима версия node.js не ниже 14.
|
||||||
|
|
||||||
```
|
```
|
||||||
$ git clone https://github.com/bookpauk/liberama
|
$ git clone https://github.com/bookpauk/liberama
|
||||||
$ cd liberama
|
$ cd liberama
|
||||||
$ npm i
|
$ npm i
|
||||||
```
|
```
|
||||||
|
|
||||||
### Windows
|
### Windows
|
||||||
```
|
```
|
||||||
$ npm run build:win
|
$ npm run build:win
|
||||||
```
|
```
|
||||||
|
|
||||||
### Linux
|
### Linux
|
||||||
```
|
```
|
||||||
$ npm run build:linux
|
$ npm run build:linux
|
||||||
```
|
```
|
||||||
|
|
||||||
Результат сборки будет доступен в каталоге `dist/linux|win` в виде исполнимого (standalone) файла
|
Результат сборки будет доступен в каталоге `dist/linux|win` в виде исполнимого (standalone) файла
|
||||||
|
|
||||||
### Разработка
|
### Разработка
|
||||||
```
|
```
|
||||||
$ npm run dev
|
$ npm run dev
|
||||||
```
|
```
|
||||||
|
|
||||||
## Помочь проекту
|
## Помочь проекту
|
||||||
|
|
||||||
* bitcoin: 3EbgZ7MK1UVaN38Gty5DCBtS4PknM4Ut85
|
* bitcoin: bc1q3tyumaj648pp2e69jalsez2lnt462ttc33nup9
|
||||||
* litecoin: MP39Riec4oSNB3XMjiquKoLWxbufRYNXxZ
|
* litecoin: MP39Riec4oSNB3XMjiquKoLWxbufRYNXxZ
|
||||||
* monero: 8BQPnvHcPSHM5gMQsmuypDgx9NNsYqwXKfDDuswEyF2Q2ewQSfd2pkK6ydH2wmMyq2JViZvy9DQ35hLMx7g72mFWNJTPtnz
|
* monero: 8BQPnvHcPSHM5gMQsmuypDgx9NNsYqwXKfDDuswEyF2Q2ewQSfd2pkK6ydH2wmMyq2JViZvy9DQ35hLMx7g72mFWNJTPtnz
|
||||||
|
|||||||
@@ -23,24 +23,6 @@ async function main() {
|
|||||||
|
|
||||||
await fs.ensureDir(tempDownloadDir);
|
await fs.ensureDir(tempDownloadDir);
|
||||||
|
|
||||||
//sqlite3
|
|
||||||
const sqliteRemoteUrl = 'https://mapbox-node-binary.s3.amazonaws.com/sqlite3/v5.0.2/napi-v3-linux-x64.tar.gz';
|
|
||||||
const sqliteDecompressedFilename = `${tempDownloadDir}/napi-v3-linux-x64/node_sqlite3.node`;
|
|
||||||
|
|
||||||
if (!await fs.pathExists(sqliteDecompressedFilename)) {
|
|
||||||
// Скачиваем node_sqlite3.node для винды, т.к. pkg не включает его в сборку
|
|
||||||
const res = await axios.get(sqliteRemoteUrl, {responseType: 'stream'})
|
|
||||||
await pipeline(res.data, fs.createWriteStream(`${tempDownloadDir}/sqlite.tar.gz`));
|
|
||||||
console.log(`done downloading ${sqliteRemoteUrl}`);
|
|
||||||
|
|
||||||
//распаковываем
|
|
||||||
console.log(await decomp.unpackTarZZ(`${tempDownloadDir}/sqlite.tar.gz`, tempDownloadDir));
|
|
||||||
console.log('files decompressed');
|
|
||||||
}
|
|
||||||
// копируем в дистрибутив
|
|
||||||
await fs.copy(sqliteDecompressedFilename, `${outDir}/node_sqlite3.node`);
|
|
||||||
console.log(`copied ${sqliteDecompressedFilename} to ${outDir}/node_sqlite3.node`);
|
|
||||||
|
|
||||||
//ipfs
|
//ipfs
|
||||||
const ipfsDecompressedFilename = `${tempDownloadDir}/go-ipfs/ipfs`;
|
const ipfsDecompressedFilename = `${tempDownloadDir}/go-ipfs/ipfs`;
|
||||||
if (!await fs.pathExists(ipfsDecompressedFilename)) {
|
if (!await fs.pathExists(ipfsDecompressedFilename)) {
|
||||||
|
|||||||
18
build/win.js
18
build/win.js
@@ -23,24 +23,6 @@ async function main() {
|
|||||||
|
|
||||||
await fs.ensureDir(tempDownloadDir);
|
await fs.ensureDir(tempDownloadDir);
|
||||||
|
|
||||||
//sqlite3
|
|
||||||
const sqliteRemoteUrl = 'https://mapbox-node-binary.s3.amazonaws.com/sqlite3/v5.0.2/napi-v3-win32-x64.tar.gz';
|
|
||||||
const sqliteDecompressedFilename = `${tempDownloadDir}/napi-v3-win32-x64/node_sqlite3.node`;
|
|
||||||
|
|
||||||
if (!await fs.pathExists(sqliteDecompressedFilename)) {
|
|
||||||
// Скачиваем node_sqlite3.node для винды, т.к. pkg не включает его в сборку
|
|
||||||
const res = await axios.get(sqliteRemoteUrl, {responseType: 'stream'})
|
|
||||||
await pipeline(res.data, fs.createWriteStream(`${tempDownloadDir}/sqlite.tar.gz`));
|
|
||||||
console.log(`done downloading ${sqliteRemoteUrl}`);
|
|
||||||
|
|
||||||
//распаковываем
|
|
||||||
console.log(await decomp.unpackTarZZ(`${tempDownloadDir}/sqlite.tar.gz`, tempDownloadDir));
|
|
||||||
console.log('files decompressed');
|
|
||||||
}
|
|
||||||
// копируем в дистрибутив
|
|
||||||
await fs.copy(sqliteDecompressedFilename, `${outDir}/node_sqlite3.node`);
|
|
||||||
console.log(`copied ${sqliteDecompressedFilename} to ${outDir}/node_sqlite3.node`);
|
|
||||||
|
|
||||||
//ipfs
|
//ipfs
|
||||||
const ipfsDecompressedFilename = `${tempDownloadDir}/go-ipfs/ipfs.exe`;
|
const ipfsDecompressedFilename = `${tempDownloadDir}/go-ipfs/ipfs.exe`;
|
||||||
if (!await fs.pathExists(ipfsDecompressedFilename)) {
|
if (!await fs.pathExists(ipfsDecompressedFilename)) {
|
||||||
|
|||||||
@@ -9,7 +9,7 @@ class Misc {
|
|||||||
async loadConfig() {
|
async loadConfig() {
|
||||||
|
|
||||||
const query = {params: [
|
const query = {params: [
|
||||||
'name', 'version', 'mode', 'maxUploadFileSize', 'useExternalBookConverter', 'acceptFileExt', 'branch',
|
'name', 'version', 'mode', 'maxUploadFileSize', 'useExternalBookConverter', 'acceptFileExt', 'bucEnabled', 'branch',
|
||||||
]};
|
]};
|
||||||
|
|
||||||
try {
|
try {
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
import axios from 'axios';
|
import axios from 'axios';
|
||||||
import * as utils from '../share/utils';
|
import * as utils from '../share/utils';
|
||||||
|
import * as cryptoUtils from '../share/cryptoUtils';
|
||||||
import wsc from './webSocketConnection';
|
import wsc from './webSocketConnection';
|
||||||
|
|
||||||
const api = axios.create({
|
const api = axios.create({
|
||||||
@@ -119,32 +120,7 @@ class Reader {
|
|||||||
estSize = response.headers['content-length'];
|
estSize = response.headers['content-length'];
|
||||||
}
|
}
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
//восстановим при необходимости файл на сервере из удаленного облака
|
//
|
||||||
let response = null
|
|
||||||
|
|
||||||
try {
|
|
||||||
response = await wsc.message(await wsc.send({action: 'reader-restore-cached-file', path: url}));
|
|
||||||
} catch (e) {
|
|
||||||
console.error(e);
|
|
||||||
//если с WebSocket проблема, работаем по http
|
|
||||||
response = await api.post('/restore-cached-file', {path: url});
|
|
||||||
response = response.data;
|
|
||||||
}
|
|
||||||
if (response.state == 'error') {
|
|
||||||
throw new Error(response.error);
|
|
||||||
}
|
|
||||||
|
|
||||||
const workerId = response.workerId;
|
|
||||||
if (!workerId)
|
|
||||||
throw new Error('Неверный ответ api');
|
|
||||||
|
|
||||||
response = await this.getWorkerStateFinish(workerId);
|
|
||||||
if (response.state == 'error') {
|
|
||||||
throw new Error(response.error);
|
|
||||||
}
|
|
||||||
if (response.size && estSize < 0) {
|
|
||||||
estSize = response.size;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return estSize;
|
return estSize;
|
||||||
@@ -174,11 +150,10 @@ class Reader {
|
|||||||
return await axios.get(url, options);
|
return await axios.get(url, options);
|
||||||
}
|
}
|
||||||
|
|
||||||
async uploadFile(file, maxUploadFileSize, callback) {
|
async uploadFile(file, maxUploadFileSize = 10*1024*1024, callback) {
|
||||||
if (!maxUploadFileSize)
|
|
||||||
maxUploadFileSize = 10*1024*1024;
|
|
||||||
if (file.size > maxUploadFileSize)
|
if (file.size > maxUploadFileSize)
|
||||||
throw new Error(`Размер файла превышает ${maxUploadFileSize} байт`);
|
throw new Error(`Размер файла превышает ${maxUploadFileSize} байт`);
|
||||||
|
|
||||||
let formData = new FormData();
|
let formData = new FormData();
|
||||||
formData.append('file', file, file.name);
|
formData.append('file', file, file.name);
|
||||||
|
|
||||||
@@ -225,6 +200,46 @@ class Reader {
|
|||||||
|
|
||||||
return response;
|
return response;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
makeUrlFromBuf(buf) {
|
||||||
|
const key = utils.toHex(cryptoUtils.sha256(buf));
|
||||||
|
return `disk://${key}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
async uploadFileBuf(buf, url) {
|
||||||
|
if (!url)
|
||||||
|
url = this.makeUrlFromBuf(buf);
|
||||||
|
|
||||||
|
let response;
|
||||||
|
try {
|
||||||
|
await axios.head(url.replace('disk://', '/upload/'), {headers: {'Cache-Control': 'no-cache'}});
|
||||||
|
response = await wsc.message(await wsc.send({action: 'upload-file-touch', url}));
|
||||||
|
} catch (e) {
|
||||||
|
response = await wsc.message(await wsc.send({action: 'upload-file-buf', buf}));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (response.error)
|
||||||
|
throw new Error(response.error);
|
||||||
|
|
||||||
|
return response;
|
||||||
|
}
|
||||||
|
|
||||||
|
async getUploadedFileBuf(url) {
|
||||||
|
url = url.replace('disk://', '/upload/');
|
||||||
|
return (await axios.get(url)).data;
|
||||||
|
}
|
||||||
|
|
||||||
|
async checkBuc(bookUrls) {
|
||||||
|
const response = await wsc.message(await wsc.send({action: 'check-buc', bookUrls}));
|
||||||
|
|
||||||
|
if (response.error)
|
||||||
|
throw new Error(response.error);
|
||||||
|
|
||||||
|
if (!response.data)
|
||||||
|
throw new Error(`response.data is empty`);
|
||||||
|
|
||||||
|
return response.data;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export default new Reader();
|
export default new Reader();
|
||||||
@@ -238,7 +238,7 @@ class App {
|
|||||||
const url = s[1] || '';
|
const url = s[1] || '';
|
||||||
const q = utils.parseQuery(s[0] || '');
|
const q = utils.parseQuery(s[0] || '');
|
||||||
if (url) {
|
if (url) {
|
||||||
q.url = decodeURIComponent(url);
|
q.url = url;
|
||||||
}
|
}
|
||||||
|
|
||||||
window.history.replaceState({}, '', '/');
|
window.history.replaceState({}, '', '/');
|
||||||
|
|||||||
@@ -1,70 +1,17 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="page">
|
<div class="page">
|
||||||
<div class="box">
|
<div class="column items-center" style="width: 500px">
|
||||||
<p class="p">
|
<p class="p">
|
||||||
Вы можете пожертвовать на развитие проекта любую сумму:
|
Здесь вы можете пожертвовать на развитие проекта:
|
||||||
</p>
|
</p>
|
||||||
<div class="address">
|
|
||||||
<img class="logo" src="./assets/yoomoney.png">
|
|
||||||
<q-btn class="q-ml-sm q-px-sm" dense no-caps @click="donateYooMoney">
|
|
||||||
Пожертвовать
|
|
||||||
</q-btn><br>
|
|
||||||
<div class="para">
|
|
||||||
{{ yooAddress }}
|
|
||||||
<q-icon class="copy-icon" name="la la-copy" @click="copyAddress(yooAddress, 'Кошелёк ЮMoney')">
|
|
||||||
<q-tooltip :delay="1000" anchor="top middle" self="center middle" content-style="font-size: 80%">
|
|
||||||
Скопировать
|
|
||||||
</q-tooltip>
|
|
||||||
</q-icon>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<!--div class="address">
|
<q-btn no-caps class="q-my-lg" color="green-8" size="14px" style="width: 200px" @click="makeDonation">
|
||||||
<img class="logo" src="./assets/paypal.png">
|
<q-icon class="q-mr-xs" name="la la-donate" size="24px" />
|
||||||
<div class="para">
|
Поддержать проект
|
||||||
{{ paypalAddress }}
|
</q-btn>
|
||||||
<q-icon class="copy-icon" name="la la-copy" @click="copyAddress(paypalAddress, 'Paypal-адрес')">
|
|
||||||
<q-tooltip :delay="1000" anchor="top middle" self="center middle" content-style="font-size: 80%">
|
|
||||||
Скопировать
|
|
||||||
</q-tooltip>
|
|
||||||
</q-icon>
|
|
||||||
</div>
|
|
||||||
</div-->
|
|
||||||
|
|
||||||
<div class="address">
|
<div style="font-size: 60%">
|
||||||
<img class="logo" src="./assets/bitcoin.png">
|
* Ваш донат является подарком автору проекта
|
||||||
<div class="para">
|
|
||||||
{{ bitcoinAddress }}
|
|
||||||
<q-icon class="copy-icon" name="la la-copy" @click="copyAddress(bitcoinAddress, 'Bitcoin-адрес')">
|
|
||||||
<q-tooltip :delay="1000" anchor="top middle" self="center middle" content-style="font-size: 80%">
|
|
||||||
Скопировать
|
|
||||||
</q-tooltip>
|
|
||||||
</q-icon>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="address">
|
|
||||||
<img class="logo" src="./assets/litecoin.png">
|
|
||||||
<div class="para">
|
|
||||||
{{ litecoinAddress }}
|
|
||||||
<q-icon class="copy-icon" name="la la-copy" @click="copyAddress(litecoinAddress, 'Litecoin-адрес')">
|
|
||||||
<q-tooltip :delay="1000" anchor="top middle" self="center middle" content-style="font-size: 80%">
|
|
||||||
Скопировать
|
|
||||||
</q-tooltip>
|
|
||||||
</q-icon>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="address">
|
|
||||||
<img class="logo" src="./assets/monero.png">
|
|
||||||
<div class="para">
|
|
||||||
{{ moneroAddress }}
|
|
||||||
<q-icon class="copy-icon" name="la la-copy" @click="copyAddress(moneroAddress, 'Monero-адрес')">
|
|
||||||
<q-tooltip :delay="1000" anchor="top middle" self="center middle" content-style="font-size: 80%">
|
|
||||||
Скопировать
|
|
||||||
</q-tooltip>
|
|
||||||
</q-icon>
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -74,28 +21,14 @@
|
|||||||
//-----------------------------------------------------------------------------
|
//-----------------------------------------------------------------------------
|
||||||
import vueComponent from '../../../vueComponent.js';
|
import vueComponent from '../../../vueComponent.js';
|
||||||
|
|
||||||
import {copyTextToClipboard} from '../../../../share/utils';
|
import * as utils from '../../../../share/utils';
|
||||||
|
|
||||||
class DonateHelpPage {
|
class DonateHelpPage {
|
||||||
yooAddress = '410018702323056';
|
|
||||||
paypalAddress = 'bookpauk@gmail.com';
|
|
||||||
bitcoinAddress = '3EbgZ7MK1UVaN38Gty5DCBtS4PknM4Ut85';
|
|
||||||
litecoinAddress = 'MP39Riec4oSNB3XMjiquKoLWxbufRYNXxZ';
|
|
||||||
moneroAddress = '8BQPnvHcPSHM5gMQsmuypDgx9NNsYqwXKfDDuswEyF2Q2ewQSfd2pkK6ydH2wmMyq2JViZvy9DQ35hLMx7g72mFWNJTPtnz';
|
|
||||||
|
|
||||||
created() {
|
created() {
|
||||||
}
|
}
|
||||||
|
|
||||||
donateYooMoney() {
|
makeDonation() {
|
||||||
window.open(`https://yoomoney.ru/to/${this.yooAddress}`, '_blank');
|
utils.makeDonation();
|
||||||
}
|
|
||||||
|
|
||||||
async copyAddress(address, prefix) {
|
|
||||||
const result = await copyTextToClipboard(address);
|
|
||||||
if (result)
|
|
||||||
this.$root.notify.success(`${prefix} ${address} успешно скопирован в буфер обмена`);
|
|
||||||
else
|
|
||||||
this.$root.notify.error('Копирование не удалось');
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -116,31 +49,4 @@ export default vueComponent(DonateHelpPage);
|
|||||||
padding: 0;
|
padding: 0;
|
||||||
text-indent: 20px;
|
text-indent: 20px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.box {
|
|
||||||
max-width: 550px;
|
|
||||||
overflow-wrap: break-word;
|
|
||||||
}
|
|
||||||
|
|
||||||
.address {
|
|
||||||
padding-top: 10px;
|
|
||||||
margin-top: 20px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.para {
|
|
||||||
margin: 10px 10px 10px 40px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.logo {
|
|
||||||
width: 130px;
|
|
||||||
position: relative;
|
|
||||||
top: 10px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.copy-icon {
|
|
||||||
margin-left: 10px;
|
|
||||||
cursor: pointer;
|
|
||||||
font-size: 120%;
|
|
||||||
color: blue;
|
|
||||||
}
|
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
Binary file not shown.
|
Before Width: | Height: | Size: 17 KiB |
Binary file not shown.
|
Before Width: | Height: | Size: 8.8 KiB |
@@ -1,5 +1,5 @@
|
|||||||
<template>
|
<template>
|
||||||
<Window @close="close">
|
<Window @close="close" style="z-index: 200">
|
||||||
<template #header>
|
<template #header>
|
||||||
Справка
|
Справка
|
||||||
</template>
|
</template>
|
||||||
@@ -36,14 +36,14 @@ import CommonHelpPage from './CommonHelpPage/CommonHelpPage.vue';
|
|||||||
import HotkeysHelpPage from './HotkeysHelpPage/HotkeysHelpPage.vue';
|
import HotkeysHelpPage from './HotkeysHelpPage/HotkeysHelpPage.vue';
|
||||||
import MouseHelpPage from './MouseHelpPage/MouseHelpPage.vue';
|
import MouseHelpPage from './MouseHelpPage/MouseHelpPage.vue';
|
||||||
import VersionHistoryPage from './VersionHistoryPage/VersionHistoryPage.vue';
|
import VersionHistoryPage from './VersionHistoryPage/VersionHistoryPage.vue';
|
||||||
//import DonateHelpPage from './DonateHelpPage/DonateHelpPage.vue';
|
import DonateHelpPage from './DonateHelpPage/DonateHelpPage.vue';
|
||||||
|
|
||||||
const pages = {
|
const pages = {
|
||||||
'CommonHelpPage': CommonHelpPage,
|
'CommonHelpPage': CommonHelpPage,
|
||||||
'HotkeysHelpPage': HotkeysHelpPage,
|
'HotkeysHelpPage': HotkeysHelpPage,
|
||||||
'MouseHelpPage': MouseHelpPage,
|
'MouseHelpPage': MouseHelpPage,
|
||||||
'VersionHistoryPage': VersionHistoryPage,
|
'VersionHistoryPage': VersionHistoryPage,
|
||||||
//'DonateHelpPage': DonateHelpPage,
|
'DonateHelpPage': DonateHelpPage,
|
||||||
};
|
};
|
||||||
|
|
||||||
const tabs = [
|
const tabs = [
|
||||||
@@ -51,7 +51,7 @@ const tabs = [
|
|||||||
['MouseHelpPage', 'Мышь/тачскрин'],
|
['MouseHelpPage', 'Мышь/тачскрин'],
|
||||||
['HotkeysHelpPage', 'Клавиатура'],
|
['HotkeysHelpPage', 'Клавиатура'],
|
||||||
['VersionHistoryPage', 'История версий'],
|
['VersionHistoryPage', 'История версий'],
|
||||||
//['DonateHelpPage', 'Помочь проекту'],
|
['DonateHelpPage', 'Помочь проекту'],
|
||||||
];
|
];
|
||||||
|
|
||||||
const componentOptions = {
|
const componentOptions = {
|
||||||
@@ -80,7 +80,7 @@ class HelpPage {
|
|||||||
}
|
}
|
||||||
|
|
||||||
activateDonateHelpPage() {
|
activateDonateHelpPage() {
|
||||||
//this.selectedTab = 'DonateHelpPage';
|
this.selectedTab = 'DonateHelpPage';
|
||||||
}
|
}
|
||||||
|
|
||||||
activateVersionHistoryHelpPage() {
|
activateVersionHistoryHelpPage() {
|
||||||
|
|||||||
@@ -57,7 +57,7 @@
|
|||||||
<div class="col column justify-end items-center no-wrap overflow-hidden">
|
<div class="col column justify-end items-center no-wrap overflow-hidden">
|
||||||
<span v-if="mode == 'omnireader'" class="bottom-span clickable" @click="findBook">Найти книгу</span>
|
<span v-if="mode == 'omnireader'" class="bottom-span clickable" @click="findBook">Найти книгу</span>
|
||||||
<span class="bottom-span clickable" @click="openHelp">Справка</span>
|
<span class="bottom-span clickable" @click="openHelp">Справка</span>
|
||||||
<!--span class="bottom-span clickable" @click="openDonate">Помочь проекту</span-->
|
<span class="bottom-span clickable" @click="openDonate">Помочь проекту</span>
|
||||||
|
|
||||||
<span v-if="version == clientVersion" class="bottom-span">v{{ version }}</span>
|
<span v-if="version == clientVersion" class="bottom-span">v{{ version }}</span>
|
||||||
<span v-else class="bottom-span">Версия сервера {{ version }}, версия клиента {{ clientVersion }}, необходимо обновить страницу</span>
|
<span v-else class="bottom-span">Версия сервера {{ version }}, версия клиента {{ clientVersion }}, необходимо обновить страницу</span>
|
||||||
|
|||||||
@@ -100,6 +100,12 @@
|
|||||||
</q-tooltip>
|
</q-tooltip>
|
||||||
</button>
|
</button>
|
||||||
<button v-show="showToolButton['recentBooks']" ref="recentBooks" v-ripple class="tool-button" :class="buttonActiveClass('recentBooks')" @click="buttonClick('recentBooks')">
|
<button v-show="showToolButton['recentBooks']" ref="recentBooks" v-ripple class="tool-button" :class="buttonActiveClass('recentBooks')" @click="buttonClick('recentBooks')">
|
||||||
|
<div v-show="bothBucEnabled && needBookUpdateCount > 0" style="position: absolute">
|
||||||
|
<div class="need-book-update-count">
|
||||||
|
{{ needBookUpdateCount }}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
<q-icon name="la la-book-open" size="32px" />
|
<q-icon name="la la-book-open" size="32px" />
|
||||||
<q-tooltip :delay="1500" anchor="bottom middle" content-style="font-size: 80%">
|
<q-tooltip :delay="1500" anchor="bottom middle" content-style="font-size: 80%">
|
||||||
{{ rstore.readerActions['recentBooks'] }}
|
{{ rstore.readerActions['recentBooks'] }}
|
||||||
@@ -156,7 +162,7 @@
|
|||||||
></SearchPage>
|
></SearchPage>
|
||||||
<CopyTextPage v-if="copyTextActive" ref="copyTextPage" @do-action="doAction"></CopyTextPage>
|
<CopyTextPage v-if="copyTextActive" ref="copyTextPage" @do-action="doAction"></CopyTextPage>
|
||||||
<LibsPage v-show="hidden" ref="libsPage" @load-book="loadBook" @libs-close="libsClose" @do-action="doAction"></LibsPage>
|
<LibsPage v-show="hidden" ref="libsPage" @load-book="loadBook" @libs-close="libsClose" @do-action="doAction"></LibsPage>
|
||||||
<RecentBooksPage v-show="recentBooksActive" ref="recentBooksPage" @load-book="loadBook" @recent-books-close="recentBooksClose"></RecentBooksPage>
|
<RecentBooksPage v-show="recentBooksActive" ref="recentBooksPage" @load-book="loadBook" @recent-books-close="recentBooksClose" @update-count-changed="updateCountChanged"></RecentBooksPage>
|
||||||
<SettingsPage v-show="settingsActive" ref="settingsPage" @do-action="doAction"></SettingsPage>
|
<SettingsPage v-show="settingsActive" ref="settingsPage" @do-action="doAction"></SettingsPage>
|
||||||
<HelpPage v-if="helpActive" ref="helpPage" @do-action="doAction"></HelpPage>
|
<HelpPage v-if="helpActive" ref="helpPage" @do-action="doAction"></HelpPage>
|
||||||
<ClickMapPage v-show="clickMapActive" ref="clickMapPage"></ClickMapPage>
|
<ClickMapPage v-show="clickMapActive" ref="clickMapPage"></ClickMapPage>
|
||||||
@@ -194,6 +200,7 @@ import ReaderDialogs from './ReaderDialogs/ReaderDialogs.vue';
|
|||||||
|
|
||||||
import bookManager from './share/bookManager';
|
import bookManager from './share/bookManager';
|
||||||
import wallpaperStorage from './share/wallpaperStorage';
|
import wallpaperStorage from './share/wallpaperStorage';
|
||||||
|
import coversStorage from './share/coversStorage';
|
||||||
import dynamicCss from '../../share/dynamicCss';
|
import dynamicCss from '../../share/dynamicCss';
|
||||||
|
|
||||||
import rstore from '../../store/modules/reader';
|
import rstore from '../../store/modules/reader';
|
||||||
@@ -308,6 +315,10 @@ class Reader {
|
|||||||
donationVisible = false;
|
donationVisible = false;
|
||||||
dualPageMode = false;
|
dualPageMode = false;
|
||||||
|
|
||||||
|
bucEnabled = false;
|
||||||
|
bucSetOnNew = false;
|
||||||
|
needBookUpdateCount = 0;
|
||||||
|
|
||||||
created() {
|
created() {
|
||||||
this.rstore = rstore;
|
this.rstore = rstore;
|
||||||
this.loading = true;
|
this.loading = true;
|
||||||
@@ -356,6 +367,32 @@ class Reader {
|
|||||||
}
|
}
|
||||||
}, 200);
|
}, 200);
|
||||||
|
|
||||||
|
this.debouncedRecentBooksPageUpdate = _.debounce(async() => {
|
||||||
|
if (this.recentBooksActive) {
|
||||||
|
await this.$refs.recentBooksPage.updateTableData();
|
||||||
|
}
|
||||||
|
}, 100);
|
||||||
|
|
||||||
|
this.recentItemKeys = [];
|
||||||
|
this.debouncedSaveRecent = _.debounce(async() => {
|
||||||
|
let timer = setTimeout(() => {
|
||||||
|
if (!this.offlineModeActive)
|
||||||
|
this.$root.notify.error('Таймаут соединения');
|
||||||
|
}, 10000);
|
||||||
|
|
||||||
|
try {
|
||||||
|
const itemKeys = this.recentItemKeys;
|
||||||
|
this.recentItemKeys = [];
|
||||||
|
//сохранение в удаленном хранилище
|
||||||
|
await this.$refs.serverStorage.saveRecent(itemKeys);
|
||||||
|
} catch (e) {
|
||||||
|
if (!this.offlineModeActive)
|
||||||
|
this.$root.notify.error(e.message);
|
||||||
|
} finally {
|
||||||
|
clearTimeout(timer);
|
||||||
|
}
|
||||||
|
}, 500, {maxWait: 1000});
|
||||||
|
|
||||||
document.addEventListener('fullscreenchange', () => {
|
document.addEventListener('fullscreenchange', () => {
|
||||||
this.fullScreenActive = (document.fullscreenElement !== null);
|
this.fullScreenActive = (document.fullscreenElement !== null);
|
||||||
});
|
});
|
||||||
@@ -366,6 +403,8 @@ class Reader {
|
|||||||
mounted() {
|
mounted() {
|
||||||
(async() => {
|
(async() => {
|
||||||
await wallpaperStorage.init();
|
await wallpaperStorage.init();
|
||||||
|
await coversStorage.init();
|
||||||
|
|
||||||
await bookManager.init(this.settings);
|
await bookManager.init(this.settings);
|
||||||
bookManager.addEventListener(this.bookManagerEvent);
|
bookManager.addEventListener(this.bookManagerEvent);
|
||||||
|
|
||||||
@@ -391,16 +430,30 @@ class Reader {
|
|||||||
this.updateRoute();
|
this.updateRoute();
|
||||||
|
|
||||||
await this.$refs.dialogs.init();
|
await this.$refs.dialogs.init();
|
||||||
|
|
||||||
|
this.$refs.recentBooksPage.init();
|
||||||
})();
|
})();
|
||||||
|
|
||||||
|
//проверки обновлений читалки
|
||||||
(async() => {
|
(async() => {
|
||||||
this.isFirstNeedUpdateNotify = true;
|
this.isFirstNeedUpdateNotify = true;
|
||||||
//вечный цикл, запрашиваем периодически конфиг для проверки выхода новой версии читалки
|
//вечный цикл, запрашиваем периодически конфиг для проверки выхода новой версии читалки
|
||||||
while (true) {// eslint-disable-line no-constant-condition
|
while (1) {// eslint-disable-line no-constant-condition
|
||||||
await this.checkNewVersionAvailable();
|
await this.checkNewVersionAvailable();
|
||||||
await utils.sleep(3600*1000); //каждый час
|
await utils.sleep(60*60*1000); //каждый час
|
||||||
}
|
}
|
||||||
//дальше кода нет
|
//дальше хода нет
|
||||||
|
})();
|
||||||
|
|
||||||
|
//проверки обновлений книг
|
||||||
|
(async() => {
|
||||||
|
await utils.sleep(15*1000); //подождем неск. секунд перед первым запросом
|
||||||
|
//вечный цикл, запрашиваем периодически обновления
|
||||||
|
while (1) {// eslint-disable-line no-constant-condition
|
||||||
|
await this.checkBuc();
|
||||||
|
await utils.sleep(70*60*1000); //каждые 70 минут
|
||||||
|
}
|
||||||
|
//дальше хода нет
|
||||||
})();
|
})();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -422,6 +475,11 @@ class Reader {
|
|||||||
this.pdfQuality = settings.pdfQuality;
|
this.pdfQuality = settings.pdfQuality;
|
||||||
this.dualPageMode = settings.dualPageMode;
|
this.dualPageMode = settings.dualPageMode;
|
||||||
this.userWallpapers = settings.userWallpapers;
|
this.userWallpapers = settings.userWallpapers;
|
||||||
|
this.bucEnabled = settings.bucEnabled;
|
||||||
|
this.bucSizeDiff = settings.bucSizeDiff;
|
||||||
|
this.bucSetOnNew = settings.bucSetOnNew;
|
||||||
|
this.bucCancelEnabled = settings.bucCancelEnabled;
|
||||||
|
this.bucCancelDays = settings.bucCancelDays;
|
||||||
|
|
||||||
this.readerActionByKeyCode = utils.userHotKeysObjectSwap(settings.userHotKeys);
|
this.readerActionByKeyCode = utils.userHotKeysObjectSwap(settings.userHotKeys);
|
||||||
this.$root.readerActionByKeyEvent = (event) => {
|
this.$root.readerActionByKeyEvent = (event) => {
|
||||||
@@ -450,22 +508,47 @@ class Reader {
|
|||||||
|
|
||||||
//wallpaper css
|
//wallpaper css
|
||||||
async loadWallpapers() {
|
async loadWallpapers() {
|
||||||
const wallpaperDataLength = await wallpaperStorage.getLength();
|
if (!_.isEqual(this.userWallpapers, this.prevUserWallpapers)) {//оптимизация
|
||||||
if (wallpaperDataLength !== this.wallpaperDataLength) {//оптимизация
|
this.prevUserWallpapers = _.cloneDeep(this.userWallpapers);
|
||||||
this.wallpaperDataLength = wallpaperDataLength;
|
|
||||||
|
|
||||||
let newCss = '';
|
let newCss = '';
|
||||||
|
let updated = false;
|
||||||
|
const wallpaperExists = new Set();
|
||||||
for (const wp of this.userWallpapers) {
|
for (const wp of this.userWallpapers) {
|
||||||
const data = await wallpaperStorage.getData(wp.cssClass);
|
wallpaperExists.add(wp.cssClass);
|
||||||
|
|
||||||
|
let data = await wallpaperStorage.getData(wp.cssClass);
|
||||||
if (!data) {
|
if (!data) {
|
||||||
//здесь будем восстанавливать данные с сервера
|
//здесь будем восстанавливать данные с сервера
|
||||||
|
const url = `disk://${wp.cssClass.replace('user-paper', '')}`;
|
||||||
|
try {
|
||||||
|
data = await readerApi.getUploadedFileBuf(url);
|
||||||
|
await wallpaperStorage.setData(wp.cssClass, data);
|
||||||
|
updated = true;
|
||||||
|
} catch (e) {
|
||||||
|
console.error(e);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (data) {
|
if (data) {
|
||||||
newCss += `.${wp.cssClass} {background: url(${data}) center; background-size: 100% 100%;}`;
|
newCss += `.${wp.cssClass} {background: url(${data}) center; background-size: 100% 100%;}`;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//почистим wallpaperStorage
|
||||||
|
for (const key of await wallpaperStorage.getKeys()) {
|
||||||
|
if (!wallpaperExists.has(key)) {
|
||||||
|
await wallpaperStorage.removeData(key);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//обновим settings, если загружали обои из /upload/
|
||||||
|
if (updated) {
|
||||||
|
const newSettings = _.cloneDeep(this.settings);
|
||||||
|
newSettings.needUpdateSettingsView = (newSettings.needUpdateSettingsView < 10 ? newSettings.needUpdateSettingsView + 1 : 0);
|
||||||
|
this.commit('reader/setSettings', newSettings);
|
||||||
|
}
|
||||||
|
|
||||||
dynamicCss.replace('wallpapers', newCss);
|
dynamicCss.replace('wallpapers', newCss);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -494,6 +577,92 @@ class Reader {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async checkBuc() {
|
||||||
|
if (!this.bothBucEnabled)
|
||||||
|
return;
|
||||||
|
|
||||||
|
try {
|
||||||
|
const sorted = bookManager.getSortedRecent();
|
||||||
|
|
||||||
|
//выберем все кандидиаты на обновление
|
||||||
|
const updateUrls = new Set();
|
||||||
|
for (const book of sorted) {
|
||||||
|
if (!book.deleted && book.checkBuc && book.url && book.url.indexOf('disk://') !== 0)
|
||||||
|
updateUrls.add(book.url);
|
||||||
|
}
|
||||||
|
|
||||||
|
//теперь по кусочкам запросим сервер
|
||||||
|
const arr = Array.from(updateUrls);
|
||||||
|
const bucSize = {};
|
||||||
|
const chunkSize = 100;
|
||||||
|
for (let i = 0; i < arr.length; i += chunkSize) {
|
||||||
|
const chunk = arr.slice(i, i + chunkSize);
|
||||||
|
|
||||||
|
const data = await readerApi.checkBuc(chunk);
|
||||||
|
|
||||||
|
for (const item of data) {
|
||||||
|
bucSize[item.id] = item.size;
|
||||||
|
}
|
||||||
|
|
||||||
|
await utils.sleep(1000);//чтобы не ддосить сервер
|
||||||
|
}
|
||||||
|
|
||||||
|
const checkSetTime = {};
|
||||||
|
//проставим новые размеры у книг
|
||||||
|
for (const book of sorted) {
|
||||||
|
if (book.deleted)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
//размер 0 считаем отсутствующим
|
||||||
|
if (book.url && bucSize[book.url] && bucSize[book.url] !== book.bucSize) {
|
||||||
|
book.bucSize = bucSize[book.url];
|
||||||
|
await bookManager.recentSetItem(book);
|
||||||
|
}
|
||||||
|
|
||||||
|
//подготовка к следующему шагу, ищем книгу по url с максимальной датой установки checkBucTime/loadTime
|
||||||
|
//от этой даты будем потом отсчитывать bucCancelDays
|
||||||
|
if (updateUrls.has(book.url)) {
|
||||||
|
let rec = checkSetTime[book.url] || {time: 0, loadTime: 0};
|
||||||
|
|
||||||
|
const time = (book.checkBucTime ? book.checkBucTime : (rec.loadTime || 0));
|
||||||
|
if (time > rec.time || (time == rec.time && (book.loadTime > rec.loadTime)))
|
||||||
|
rec = {time, loadTime: book.loadTime, key: book.key};
|
||||||
|
|
||||||
|
checkSetTime[book.url] = rec;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//bucCancelEnabled и bucCancelDays
|
||||||
|
//снимем флаг checkBuc у необновлявшихся bucCancelDays
|
||||||
|
if (this.bucCancelEnabled) {
|
||||||
|
for (const rec of Object.values(checkSetTime)) {
|
||||||
|
if (rec.time && Date.now() - rec.time > this.bucCancelDays*24*3600*1000) {
|
||||||
|
const book = await bookManager.getRecentBook({key: rec.key});
|
||||||
|
const needBookUpdate =
|
||||||
|
book.checkBuc
|
||||||
|
&& book.bucSize
|
||||||
|
&& utils.hasProp(book, 'downloadSize')
|
||||||
|
&& book.bucSize !== book.downloadSize
|
||||||
|
&& (book.bucSize - book.downloadSize >= this.bucSizeDiff)
|
||||||
|
;
|
||||||
|
|
||||||
|
if (book && !needBookUpdate) {
|
||||||
|
await bookManager.setCheckBuc(book, undefined);//!!!
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
await this.$refs.recentBooksPage.updateTableData();
|
||||||
|
} catch (e) {
|
||||||
|
console.error(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
updateCountChanged(event) {
|
||||||
|
this.needBookUpdateCount = event.needBookUpdateCount;
|
||||||
|
}
|
||||||
|
|
||||||
checkSetStorageAccessKey() {
|
checkSetStorageAccessKey() {
|
||||||
const q = this.$route.query;
|
const q = this.$route.query;
|
||||||
|
|
||||||
@@ -552,7 +721,7 @@ class Reader {
|
|||||||
return;
|
return;
|
||||||
const recent = this.mostRecentBook();
|
const recent = this.mostRecentBook();
|
||||||
const pos = (recent && recent.bookPos && this.allowUrlParamBookPos ? `__p=${recent.bookPos}&` : '');
|
const pos = (recent && recent.bookPos && this.allowUrlParamBookPos ? `__p=${recent.bookPos}&` : '');
|
||||||
const url = (recent ? `url=${recent.url}` : '');
|
const url = (recent ? `url=${encodeURIComponent(recent.url)}` : '');
|
||||||
if (isNewRoute)
|
if (isNewRoute)
|
||||||
this.$router.push(`/reader?${pos}${url}`).catch(() => {});
|
this.$router.push(`/reader?${pos}${url}`).catch(() => {});
|
||||||
else
|
else
|
||||||
@@ -572,6 +741,10 @@ class Reader {
|
|||||||
return versionHistory[0].version;
|
return versionHistory[0].version;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
get bothBucEnabled() {
|
||||||
|
return this.$store.state.config.bucEnabled && this.bucEnabled;
|
||||||
|
}
|
||||||
|
|
||||||
get routeParamUrl() {
|
get routeParamUrl() {
|
||||||
let result = '';
|
let result = '';
|
||||||
const path = this.$route.fullPath;
|
const path = this.$route.fullPath;
|
||||||
@@ -620,27 +793,12 @@ class Reader {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (eventName == 'recent-changed') {
|
if (eventName == 'recent-changed') {
|
||||||
if (this.recentBooksActive) {
|
this.debouncedRecentBooksPageUpdate();
|
||||||
await this.$refs.recentBooksPage.updateTableData();
|
|
||||||
}
|
|
||||||
|
|
||||||
//сохранение в serverStorage
|
//сохранение в serverStorage
|
||||||
if (value) {
|
if (value && this.recentItemKeys.indexOf(value) < 0) {
|
||||||
await utils.sleep(500);
|
this.recentItemKeys.push(value);
|
||||||
|
this.debouncedSaveRecent();
|
||||||
let timer = setTimeout(() => {
|
|
||||||
if (!this.offlineModeActive)
|
|
||||||
this.$root.notify.error('Таймаут соединения');
|
|
||||||
}, 10000);
|
|
||||||
|
|
||||||
try {
|
|
||||||
await this.$refs.serverStorage.saveRecent(value);
|
|
||||||
} catch (e) {
|
|
||||||
if (!this.offlineModeActive)
|
|
||||||
this.$root.notify.error(e.message);
|
|
||||||
} finally {
|
|
||||||
clearTimeout(timer);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -998,7 +1156,6 @@ class Reader {
|
|||||||
classResult = classDisabled;
|
classResult = classDisabled;
|
||||||
break;
|
break;
|
||||||
case 'refresh':
|
case 'refresh':
|
||||||
case 'recentBooks':
|
|
||||||
if (!this.mostRecentBookReactive)
|
if (!this.mostRecentBookReactive)
|
||||||
classResult = classDisabled;
|
classResult = classDisabled;
|
||||||
break;
|
break;
|
||||||
@@ -1108,6 +1265,7 @@ class Reader {
|
|||||||
wasOpened = (wasOpened ? _.cloneDeep(wasOpened) : {});
|
wasOpened = (wasOpened ? _.cloneDeep(wasOpened) : {});
|
||||||
|
|
||||||
wasOpened = Object.assign(wasOpened, {
|
wasOpened = Object.assign(wasOpened, {
|
||||||
|
url: (opts.url !== undefined ? opts.url : wasOpened.url),
|
||||||
path: (opts.path !== undefined ? opts.path : wasOpened.path),
|
path: (opts.path !== undefined ? opts.path : wasOpened.path),
|
||||||
bookPos: (opts.bookPos !== undefined ? opts.bookPos : wasOpened.bookPos),
|
bookPos: (opts.bookPos !== undefined ? opts.bookPos : wasOpened.bookPos),
|
||||||
bookPosSeen: (opts.bookPos !== undefined ? opts.bookPos : wasOpened.bookPosSeen),
|
bookPosSeen: (opts.bookPos !== undefined ? opts.bookPos : wasOpened.bookPosSeen),
|
||||||
@@ -1133,6 +1291,7 @@ class Reader {
|
|||||||
|
|
||||||
this.checkBookPosPercent();
|
this.checkBookPosPercent();
|
||||||
this.activateClickMapPage();//no await
|
this.activateClickMapPage();//no await
|
||||||
|
this.$refs.recentBooksPage.updateTableData();//no await
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1209,9 +1368,13 @@ class Reader {
|
|||||||
delete wasOpened.loadTime;
|
delete wasOpened.loadTime;
|
||||||
|
|
||||||
// добавляем в историю
|
// добавляем в историю
|
||||||
await bookManager.setRecentBook(Object.assign(wasOpened, addedBook));
|
const recentBook = await bookManager.setRecentBook(Object.assign(wasOpened, addedBook));
|
||||||
|
if (this.bucSetOnNew) {
|
||||||
|
await bookManager.setCheckBuc(recentBook, true);
|
||||||
|
}
|
||||||
|
|
||||||
this.mostRecentBook();
|
this.mostRecentBook();
|
||||||
this.addAction(wasOpened.bookPos);
|
this.addAction(recentBook.bookPos);
|
||||||
this.updateRoute(true);
|
this.updateRoute(true);
|
||||||
|
|
||||||
this.loaderActive = false;
|
this.loaderActive = false;
|
||||||
@@ -1223,6 +1386,7 @@ class Reader {
|
|||||||
|
|
||||||
this.checkBookPosPercent();
|
this.checkBookPosPercent();
|
||||||
this.activateClickMapPage();//no await
|
this.activateClickMapPage();//no await
|
||||||
|
this.$refs.recentBooksPage.updateTableData();//no await
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
progress.hide(); this.progressActive = false;
|
progress.hide(); this.progressActive = false;
|
||||||
this.loaderActive = true;
|
this.loaderActive = true;
|
||||||
@@ -1573,4 +1737,16 @@ export default vueComponent(Reader);
|
|||||||
.clear {
|
.clear {
|
||||||
color: rgba(0,0,0,0);
|
color: rgba(0,0,0,0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.need-book-update-count {
|
||||||
|
position: relative;
|
||||||
|
padding: 2px 6px 2px 6px;
|
||||||
|
left: 27px;
|
||||||
|
top: 22px;
|
||||||
|
background-color: blue;
|
||||||
|
border-radius: 10px;
|
||||||
|
color: white;
|
||||||
|
z-index: 10;
|
||||||
|
font-size: 80%;
|
||||||
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -18,56 +18,51 @@
|
|||||||
</template>
|
</template>
|
||||||
</Dialog>
|
</Dialog>
|
||||||
|
|
||||||
<Dialog ref="dialog2" v-model="donationVisible">
|
<q-dialog ref="dialog2" v-model="donationVisible" style="z-index: 100" no-route-dismiss no-esc-dismiss no-backdrop-dismiss>
|
||||||
<template #header>
|
<div class="column bg-white no-wrap q-pa-md">
|
||||||
Здравствуйте, уважаемые читатели!
|
<div class="row justify-center q-mb-md" style="font-size: 110%">
|
||||||
</template>
|
Здравствуйте, дорогие читатели!
|
||||||
|
</div>
|
||||||
|
|
||||||
<div style="word-break: normal">
|
<div class="q-mx-md column" style="word-break: normal">
|
||||||
Стартовала ежегодная акция "Оплатим хостинг вместе".<br><br>
|
<div>
|
||||||
|
Вот уже много лет мы все вместе пользуемся нашей любимой читалкой.<br><br>
|
||||||
|
|
||||||
Для оплаты годового хостинга читалки, необходимо собрать около 2000 рублей.
|
Напоминаем вам, что проект является некоммерческим и обладает такими
|
||||||
В настоящий момент у автора эта сумма есть в наличии. Однако будет справедливо, если каждый
|
достоинствами, как:
|
||||||
сможет проголосовать рублем за то, чтобы читалка так и оставалась:
|
|
||||||
|
|
||||||
<ul>
|
<ul>
|
||||||
<li>непрерывно улучшаемой</li>
|
<li>все функции читалки открыты и доступны совершенно бесплатно</li>
|
||||||
<li>без рекламы</li>
|
<li>в проекте отсутствует какая-либо реклама или баннеры</li>
|
||||||
<li>без регистрации</li>
|
<li>нет никакой регистрации и монетизации</li>
|
||||||
<li>Open Source</li>
|
<li>нет сбора персональных данных</li>
|
||||||
</ul>
|
<li>открытый исходный код</li>
|
||||||
|
<li>проект постепенно улучшается, по мере возможности</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
Автор также обращается с просьбой о помощи в распространении
|
Однако на оплату хостинга читалки и сервера обновлений автор тратит свои
|
||||||
<a href="https://omnireader.ru" target="_blank">ссылки</a>
|
собственные средства, а также тратит свое время и силы на улучшение проекта.
|
||||||
<q-icon class="copy-icon" name="la la-copy" @click="copyLink('https://omnireader.ru')">
|
<br><br>
|
||||||
<q-tooltip :delay="1000" anchor="top middle" self="center middle" content-style="font-size: 80%">
|
Поддержим же материально наш ресурс, чтобы и дальше спокойно существовать и развиваться:
|
||||||
Скопировать
|
</div>
|
||||||
</q-tooltip>
|
|
||||||
</q-icon>
|
|
||||||
на читалку через тематические форумы, соцсети, мессенджеры и пр.
|
|
||||||
Чем нас больше, тем легче оставаться на плаву и тем больше мотивации у разработчика, чтобы продолжать работать над проектом.
|
|
||||||
|
|
||||||
<br><br>
|
<q-btn style="margin: 10px 50px 10px 50px" color="green-8" size="14px" no-caps @click="makeDonation">
|
||||||
Если соберется бóльшая сумма, то разработка децентрализованной библиотеки для свободного обмена книгами будет по возможности ускорена.
|
<q-icon class="q-mr-xs" name="la la-donate" size="24px" />
|
||||||
<br><br>
|
Поддержать проект
|
||||||
P.S. При необходимости можно воспользоваться подходящим обменником на <a href="https://www.bestchange.ru" target="_blank">bestchange.ru</a>
|
</q-btn>
|
||||||
|
|
||||||
<br><br>
|
<q-btn style="margin: 0 50px 20px 50px" size="14px" no-caps @click="donationDialogRemind">
|
||||||
<div class="row justify-center">
|
Напомнить в следующем месяце
|
||||||
<!--q-btn class="q-px-sm" color="primary" dense no-caps @click="openDonate">
|
</q-btn>
|
||||||
Помочь проекту
|
|
||||||
</q-btn-->
|
<div class="row justify-center">
|
||||||
|
<div class="q-px-sm clickable" style="font-size: 80%" @click="openDonate">
|
||||||
|
Помочь проекту можно в любое время
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
</q-dialog>
|
||||||
<template #footer>
|
|
||||||
<span class="clickable row justify-end" style="font-size: 60%; color: grey" @click="donationDialogDisable">Больше не показывать</span>
|
|
||||||
<br>
|
|
||||||
<q-btn class="q-px-sm" dense no-caps @click="donationDialogRemind">
|
|
||||||
Напомнить позже
|
|
||||||
</q-btn>
|
|
||||||
</template>
|
|
||||||
</Dialog>
|
|
||||||
|
|
||||||
<Dialog ref="dialog3" v-model="urlHelpVisible">
|
<Dialog ref="dialog3" v-model="urlHelpVisible">
|
||||||
<template #header>
|
<template #header>
|
||||||
@@ -134,7 +129,7 @@ class ReaderDialogs {
|
|||||||
loadSettings() {
|
loadSettings() {
|
||||||
const settings = this.settings;
|
const settings = this.settings;
|
||||||
this.showWhatsNewDialog = settings.showWhatsNewDialog;
|
this.showWhatsNewDialog = settings.showWhatsNewDialog;
|
||||||
this.showDonationDialog2020 = settings.showDonationDialog2020;
|
this.showDonationDialog = settings.showDonationDialog;
|
||||||
}
|
}
|
||||||
|
|
||||||
async showWhatsNew() {
|
async showWhatsNew() {
|
||||||
@@ -149,9 +144,9 @@ class ReaderDialogs {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async showDonation() {
|
async showDonation() {
|
||||||
const today = utils.formatDate(new Date(), 'coDate');
|
const today = utils.formatDate(new Date(), 'coMonth');
|
||||||
|
|
||||||
if ((this.mode == 'omnireader' || this.mode == 'liberama.top') && today < '2020-03-01' && this.showDonationDialog2020 && this.donationRemindDate != today) {
|
if ((this.mode == 'omnireader' || this.mode == 'liberama.top') && this.showDonationDialog && this.donationRemindDate != today) {
|
||||||
await utils.sleep(3000);
|
await utils.sleep(3000);
|
||||||
this.donationVisible = true;
|
this.donationVisible = true;
|
||||||
}
|
}
|
||||||
@@ -166,20 +161,17 @@ class ReaderDialogs {
|
|||||||
this.urlHelpVisible = false;
|
this.urlHelpVisible = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
donationDialogDisable() {
|
|
||||||
this.donationVisible = false;
|
|
||||||
if (this.showDonationDialog2020) {
|
|
||||||
this.commit('reader/setSettings', { showDonationDialog2020: false });
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
donationDialogRemind() {
|
donationDialogRemind() {
|
||||||
this.donationVisible = false;
|
this.donationVisible = false;
|
||||||
this.commit('reader/setDonationRemindDate', utils.formatDate(new Date(), 'coDate'));
|
this.commit('reader/setDonationRemindDate', utils.formatDate(new Date(), 'coMonth'));
|
||||||
|
}
|
||||||
|
|
||||||
|
makeDonation() {
|
||||||
|
utils.makeDonation();
|
||||||
|
this.donationDialogRemind();
|
||||||
}
|
}
|
||||||
|
|
||||||
openDonate() {
|
openDonate() {
|
||||||
this.donationVisible = false;
|
|
||||||
this.$emit('donate-toggle');
|
this.$emit('donate-toggle');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -7,6 +7,32 @@
|
|||||||
</span>
|
</span>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
|
<template #buttons>
|
||||||
|
<div
|
||||||
|
v-show="needBookUpdateCount > 0"
|
||||||
|
class="row justify-center items-center"
|
||||||
|
:class="{'header-button-update': !showNeedBookUpdateOnly, 'header-button-update-pressed': showNeedBookUpdateOnly}"
|
||||||
|
@mousedown.stop @click="showNeedBookUpdateOnlyToggle"
|
||||||
|
>
|
||||||
|
<span style="font-size: 90%">{{ needBookUpdateCount }} обновлен{{ wordEnding(needBookUpdateCount, 3) }}</span>
|
||||||
|
<q-tooltip :delay="1500" anchor="bottom middle" content-style="font-size: 80%">
|
||||||
|
{{ (needBookUpdateCount ? 'Скрыть обновления' : 'Показать обновления') }}
|
||||||
|
</q-tooltip>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div
|
||||||
|
class="row justify-center items-center"
|
||||||
|
:class="{'header-button': !showArchive, 'header-button-pressed': showArchive}"
|
||||||
|
@mousedown.stop @click="showArchiveToggle"
|
||||||
|
>
|
||||||
|
<q-icon class="q-mr-xs" name="la la-archive" size="20px" />
|
||||||
|
<span style="font-size: 90%">Архив</span>
|
||||||
|
<q-tooltip :delay="1500" anchor="bottom middle" content-style="font-size: 80%">
|
||||||
|
{{ (showArchive ? 'Скрыть архивные' : 'Показать архивные') }}
|
||||||
|
</q-tooltip>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
<a ref="download" style="display: none;" target="_blank"></a>
|
<a ref="download" style="display: none;" target="_blank"></a>
|
||||||
|
|
||||||
<div id="vs-container" ref="vsContainer" class="recent-books-scroll col">
|
<div id="vs-container" ref="vsContainer" class="recent-books-scroll col">
|
||||||
@@ -39,10 +65,25 @@
|
|||||||
</q-tooltip>
|
</q-tooltip>
|
||||||
</q-btn>
|
</q-btn>
|
||||||
|
|
||||||
|
<q-input
|
||||||
|
ref="input"
|
||||||
|
v-model="search"
|
||||||
|
class="q-ml-sm q-mt-xs"
|
||||||
|
outlined dense
|
||||||
|
style="width: 185px"
|
||||||
|
bg-color="white"
|
||||||
|
placeholder="Найти"
|
||||||
|
@click.stop
|
||||||
|
>
|
||||||
|
<template #append>
|
||||||
|
<q-icon v-if="search !== ''" name="la la-times" class="cursor-pointer" @click.stop="resetSearch" />
|
||||||
|
</template>
|
||||||
|
</q-input>
|
||||||
|
|
||||||
<q-select
|
<q-select
|
||||||
ref="sortMethod"
|
ref="sortMethod"
|
||||||
v-model="sortMethod"
|
v-model="sortMethod"
|
||||||
class="q-ml-md q-mt-xs"
|
class="q-ml-sm q-mt-xs"
|
||||||
:options="sortMethodOptions"
|
:options="sortMethodOptions"
|
||||||
style="width: 180px"
|
style="width: 180px"
|
||||||
bg-color="white"
|
bg-color="white"
|
||||||
@@ -60,21 +101,6 @@
|
|||||||
<div style="height: 28px; padding-top: 2px; overflow: hidden" v-html="scope.opt.label" />
|
<div style="height: 28px; padding-top: 2px; overflow: hidden" v-html="scope.opt.label" />
|
||||||
</template>
|
</template>
|
||||||
</q-select>
|
</q-select>
|
||||||
|
|
||||||
<q-input
|
|
||||||
ref="input"
|
|
||||||
v-model="search"
|
|
||||||
class="q-ml-sm q-mt-xs"
|
|
||||||
outlined dense
|
|
||||||
style="width: 180px"
|
|
||||||
bg-color="white"
|
|
||||||
placeholder="Найти"
|
|
||||||
@click.stop
|
|
||||||
>
|
|
||||||
<template #append>
|
|
||||||
<q-icon v-if="search !== ''" name="la la-times" class="cursor-pointer" @click.stop="resetSearch" />
|
|
||||||
</template>
|
|
||||||
</q-input>
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<q-virtual-scroll
|
<q-virtual-scroll
|
||||||
@@ -86,13 +112,22 @@
|
|||||||
@virtual-scroll="onScroll"
|
@virtual-scroll="onScroll"
|
||||||
>
|
>
|
||||||
<div class="table-row row" :class="{even: index % 2 > 0, 'active-book': item.active, 'active-parent-book': item.activeParent}">
|
<div class="table-row row" :class="{even: index % 2 > 0, 'active-book': item.active, 'active-parent-book': item.activeParent}">
|
||||||
<div v-show="item.inGroup" class="row-part column justify-center items-center" style="width: 40px; border-right: 1px solid #cccccc">
|
<div v-show="item.inGroup" class="row-part column justify-center items-center" style="width: 40px">
|
||||||
<q-icon name="la la-code-branch" size="24px" style="color: green" />
|
<q-icon name="la la-code-branch" size="24px" style="color: green" />
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="row-part column justify-center items-stretch" style="width: 80px">
|
<div class="row-part column justify-center items-stretch" style="width: 80px">
|
||||||
<div class="col row justify-center items-center clickable" @click="loadBook(item)">
|
<div class="col row justify-center items-center clickable" style="padding: 0 2px 0 2px" @click="loadBook(item, bothBucEnabled && item.needBookUpdate)">
|
||||||
<q-icon name="la la-book" size="40px" style="color: #dddddd" />
|
<div v-show="isLoadedCover(item.coverPageUrl)" style="height: 80px" v-html="getCoverHtml(item.coverPageUrl)" />
|
||||||
|
<q-icon v-show="!isLoadedCover(item.coverPageUrl)" name="la la-book" size="40px" style="color: #dddddd" />
|
||||||
|
|
||||||
|
<div
|
||||||
|
v-show="bothBucEnabled && item.needBookUpdate"
|
||||||
|
class="column justify-center"
|
||||||
|
style="position: absolute; background-color: rgba(255, 255, 255, 0.5); border-radius: 40px;"
|
||||||
|
>
|
||||||
|
<q-icon name="la la-sync" size="60px" style="color: blue" />
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div v-show="!showSameBook && item.group && item.group.length > 0" class="row justify-center" style="font-size: 70%">
|
<div v-show="!showSameBook && item.group && item.group.length > 0" class="row justify-center" style="font-size: 70%">
|
||||||
@@ -100,58 +135,111 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="row-part column items-stretch clickable break-word" :style="{ 'width': (350 - 40*(+item.inGroup)) + 'px' }" style="font-size: 75%" @click="loadBook(item)">
|
<div class="row-part column items-stretch clickable break-word" @click="loadBook(item)">
|
||||||
<div class="row" style="font-size: 80%">
|
<div
|
||||||
<div class="row justify-center row-info-top" style="width: 30px">
|
class="col" style="border: 1px solid #cccccc; border-bottom: 0; padding: 4px; line-height: 140%;"
|
||||||
{{ item.num }}
|
:style="{ 'width': (380 - 40*(+item.inGroup)) + 'px' }"
|
||||||
</div>
|
>
|
||||||
<div class="row justify-center row-info-top" style="width: 130px">
|
<div class="text-green-10" style="font-size: 80%">
|
||||||
Читался: {{ item.touchTime }}
|
|
||||||
</div>
|
|
||||||
<div class="row justify-center row-info-top" style="width: 138px">
|
|
||||||
Загружен: {{ item.loadTime }}
|
|
||||||
</div>
|
|
||||||
<div class="row justify-center row-info-top" style="width: 1px">
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="col q-mt-xs" :style="{ 'width': (340 - 40*(+item.inGroup)) + 'px' }">
|
|
||||||
<div class="text-green-10" style="font-size: 105%">
|
|
||||||
{{ item.desc.author }}
|
{{ item.desc.author }}
|
||||||
</div>
|
</div>
|
||||||
<div>{{ item.desc.title }}</div>
|
<div style="font-size: 75%">
|
||||||
<!--div>{{ item.path }}</div-->
|
{{ item.desc.title }}
|
||||||
|
</div>
|
||||||
|
<div v-show="bothBucEnabled && item.needBookUpdate" style="font-size: 75%; color: blue;">
|
||||||
|
Размер: {{ item.bucSize - item.downloadSize > 0 ? '+' : '' }}{{ item.bucSize - item.downloadSize }}
|
||||||
|
({{ item.downloadSize }} → {{ item.bucSize }})
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="row q-mt-xs" style="font-size: 80%">
|
<div class="row" style="font-size: 10px">
|
||||||
<div class="row justify-center row-info-bottom" style="width: 60px">
|
<div class="row justify-center items-center row-info-top" style="width: 60px">
|
||||||
{{ item.desc.textLen }}
|
{{ item.desc.textLen }}
|
||||||
</div>
|
</div>
|
||||||
<div class="row justify-center row-info-bottom" style="width: 60px">
|
|
||||||
|
<div class="row items-center row-info-top" :style="`width: ${(260 - 40*(+item.inGroup))}px; padding: 1px`">
|
||||||
|
<div class="read-bar" :style="`width: ${100*item.readPart}%`"></div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="row justify-center items-center row-info-top" style="width: 59px">
|
||||||
{{ item.desc.perc }}
|
{{ item.desc.perc }}
|
||||||
</div>
|
</div>
|
||||||
<div class="row justify-center row-info-bottom" style="width: 1px">
|
<div class="row-info-top" style="width: 1px">
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="read-bar" :style="`width: ${(340 - 40*(+item.inGroup))*item.readPart}px`"></div>
|
<div class="row" style="font-size: 10px" :style="{ 'width': (380 - 40*(+item.inGroup)) + 'px' }">
|
||||||
</div>
|
<div class="row justify-center items-center row-info-bottom" style="width: 30px">
|
||||||
|
{{ item.num }}
|
||||||
<div class="row-part column justify-center" style="width: 80px; font-size: 75%">
|
</div>
|
||||||
<div>
|
<div class="col row">
|
||||||
<a v-show="isUrl(item.url)" :href="item.url" target="_blank">Оригинал</a><br><br>
|
<div class="row justify-center items-center row-info-bottom time-info" style="width: 50%">
|
||||||
<a :href="item.path" @click.prevent="downloadBook(item.path, item.fullTitle)">Скачать FB2</a>
|
Загружен: {{ item.loadTime }}
|
||||||
|
</div>
|
||||||
|
<div class="row justify-center items-center row-info-bottom time-info" style="width: 50%">
|
||||||
|
Читался: {{ item.touchTime }}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="row-info-bottom" style="width: 1px">
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="row-part column justify-center">
|
<div
|
||||||
<q-btn
|
class="row-part column"
|
||||||
dense
|
style="width: 90px;"
|
||||||
style="width: 30px; height: 30px; padding: 7px 0 7px 0; margin-left: 4px"
|
>
|
||||||
|
<div
|
||||||
|
class="col column justify-center"
|
||||||
|
style="font-size: 75%; padding-left: 6px; border: 1px solid #cccccc; border-left: 0;"
|
||||||
|
>
|
||||||
|
<div style="margin: 25px 0 0 5px">
|
||||||
|
<a v-show="isUrl(item.url)" :href="item.url" target="_blank">Оригинал</a><br><br>
|
||||||
|
<a :href="item.path" @click.prevent="downloadBook(item.path, item.fullTitle)">Скачать FB2</a>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div
|
||||||
|
class="del-button self-end row justify-center items-center clickable"
|
||||||
@click="handleDel(item.key)"
|
@click="handleDel(item.key)"
|
||||||
>
|
>
|
||||||
<q-icon class="la la-times" size="14px" />
|
<q-icon class="la la-times" size="12px" />
|
||||||
</q-btn>
|
<q-tooltip :delay="1500" anchor="bottom middle" content-style="font-size: 80%">
|
||||||
|
{{ (showArchive ? 'Удалить окончательно' : 'Перенести в архив') }}
|
||||||
|
</q-tooltip>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div
|
||||||
|
v-show="showArchive"
|
||||||
|
class="restore-button self-start row justify-center items-center clickable"
|
||||||
|
@click="handleRestore(item.key)"
|
||||||
|
>
|
||||||
|
<q-icon class="la la-arrow-left" size="14px" />
|
||||||
|
<q-tooltip :delay="1500" anchor="bottom middle" content-style="font-size: 80%">
|
||||||
|
Восстановить из архива
|
||||||
|
</q-tooltip>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div
|
||||||
|
v-show="bothBucEnabled && item.showCheckBuc"
|
||||||
|
class="buc-checkbox self-start"
|
||||||
|
>
|
||||||
|
<q-checkbox
|
||||||
|
v-model="item.checkBuc"
|
||||||
|
size="xs"
|
||||||
|
style="position: relative; top: -3px; left: -3px;"
|
||||||
|
@update:model-value="checkBucChange(item)"
|
||||||
|
>
|
||||||
|
<q-tooltip :delay="1500" anchor="bottom middle" content-style="font-size: 80%">
|
||||||
|
<div v-if="item.checkBuc === undefined">
|
||||||
|
Проверка обновлений отключена автоматически<br>т.к. книга не обновлялась {{ bucCancelDays }} дней
|
||||||
|
</div>
|
||||||
|
<div v-else>
|
||||||
|
{{ (item.checkBuc ? 'Проверка обновлений книги включена' : 'Проверка обновлений книги отключена') }}
|
||||||
|
</div>
|
||||||
|
</q-tooltip>
|
||||||
|
</q-checkbox>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</q-virtual-scroll>
|
</q-virtual-scroll>
|
||||||
@@ -171,6 +259,7 @@ import LockQueue from '../../../share/LockQueue';
|
|||||||
import Window from '../../share/Window.vue';
|
import Window from '../../share/Window.vue';
|
||||||
import bookManager from '../share/bookManager';
|
import bookManager from '../share/bookManager';
|
||||||
import readerApi from '../../../api/reader';
|
import readerApi from '../../../api/reader';
|
||||||
|
import coversStorage from '../share/coversStorage';
|
||||||
|
|
||||||
const componentOptions = {
|
const componentOptions = {
|
||||||
components: {
|
components: {
|
||||||
@@ -186,6 +275,12 @@ const componentOptions = {
|
|||||||
settings() {
|
settings() {
|
||||||
this.loadSettings();
|
this.loadSettings();
|
||||||
},
|
},
|
||||||
|
needBookUpdateCount() {
|
||||||
|
if (this.needBookUpdateCount == 0)
|
||||||
|
this.showNeedBookUpdateOnly = false;
|
||||||
|
|
||||||
|
this.$emit('update-count-changed', {needBookUpdateCount: this.needBookUpdateCount});
|
||||||
|
}
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
class RecentBooksPage {
|
class RecentBooksPage {
|
||||||
@@ -196,6 +291,17 @@ class RecentBooksPage {
|
|||||||
tableData = [];
|
tableData = [];
|
||||||
sortMethod = '';
|
sortMethod = '';
|
||||||
showSameBook = false;
|
showSameBook = false;
|
||||||
|
bucEnabled = false;
|
||||||
|
bucSizeDiff = 0;
|
||||||
|
bucSetOnNew = false;
|
||||||
|
bucCancelDays = 0;
|
||||||
|
needBookUpdateCount = 0;
|
||||||
|
|
||||||
|
showArchive = false;
|
||||||
|
showNeedBookUpdateOnly = false;
|
||||||
|
|
||||||
|
covers = {};
|
||||||
|
coversLoadFunc = {};
|
||||||
|
|
||||||
created() {
|
created() {
|
||||||
this.commit = this.$store.commit;
|
this.commit = this.$store.commit;
|
||||||
@@ -221,6 +327,7 @@ class RecentBooksPage {
|
|||||||
this.showBar();
|
this.showBar();
|
||||||
await this.updateTableData();
|
await this.updateTableData();
|
||||||
await this.scrollToActiveBook();
|
await this.scrollToActiveBook();
|
||||||
|
//await this.scrollRefresh();
|
||||||
})();
|
})();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -228,12 +335,20 @@ class RecentBooksPage {
|
|||||||
const settings = this.settings;
|
const settings = this.settings;
|
||||||
this.showSameBook = settings.recentShowSameBook;
|
this.showSameBook = settings.recentShowSameBook;
|
||||||
this.sortMethod = settings.recentSortMethod || 'loadTimeDesc';
|
this.sortMethod = settings.recentSortMethod || 'loadTimeDesc';
|
||||||
|
this.bucEnabled = settings.bucEnabled;
|
||||||
|
this.bucSizeDiff = settings.bucSizeDiff;
|
||||||
|
this.bucSetOnNew = settings.bucSetOnNew;
|
||||||
|
this.bucCancelDays = settings.bucCancelDays;
|
||||||
}
|
}
|
||||||
|
|
||||||
get settings() {
|
get settings() {
|
||||||
return this.$store.state.reader.settings;
|
return this.$store.state.reader.settings;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
get bothBucEnabled() {
|
||||||
|
return this.$store.state.config.bucEnabled && this.bucEnabled;
|
||||||
|
}
|
||||||
|
|
||||||
async updateTableData() {
|
async updateTableData() {
|
||||||
if (!this.inited)
|
if (!this.inited)
|
||||||
return;
|
return;
|
||||||
@@ -247,7 +362,7 @@ class RecentBooksPage {
|
|||||||
|
|
||||||
//подготовка полей
|
//подготовка полей
|
||||||
for (const book of sorted) {
|
for (const book of sorted) {
|
||||||
if (book.deleted)
|
if ((!this.showArchive && book.deleted) || (this.showArchive && book.deleted != 1))
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
let d = new Date();
|
let d = new Date();
|
||||||
@@ -271,9 +386,14 @@ class RecentBooksPage {
|
|||||||
|
|
||||||
let title = bt.bookTitle;
|
let title = bt.bookTitle;
|
||||||
title = (title ? `"${title}"`: '');
|
title = (title ? `"${title}"`: '');
|
||||||
const author = (bt.author ? bt.author : (bt.bookTitle ? bt.bookTitle : (book.uploadFileName ? book.uploadFileName : book.url)));
|
const author = (bt.author ? bt.author : (bt.bookTitle ? bt.bookTitle : (book.uploadFileName ? book.uploadFileName : book.url))) || '';
|
||||||
|
|
||||||
result.push({
|
result.push({
|
||||||
|
key: book.key,
|
||||||
|
url: book.url,
|
||||||
|
path: book.path,
|
||||||
|
deleted: book.deleted,
|
||||||
|
|
||||||
touchTime,
|
touchTime,
|
||||||
loadTime,
|
loadTime,
|
||||||
desc: {
|
desc: {
|
||||||
@@ -283,14 +403,25 @@ class RecentBooksPage {
|
|||||||
textLen,
|
textLen,
|
||||||
},
|
},
|
||||||
readPart,
|
readPart,
|
||||||
url: book.url,
|
|
||||||
path: book.path,
|
|
||||||
fullTitle: bt.fullTitle,
|
fullTitle: bt.fullTitle,
|
||||||
key: book.key,
|
|
||||||
sameBookKey: book.sameBookKey,
|
sameBookKey: book.sameBookKey,
|
||||||
active: (activeBook.key == book.key),
|
active: (activeBook.key == book.key),
|
||||||
activeParent: false,
|
activeParent: false,
|
||||||
inGroup: false,
|
inGroup: false,
|
||||||
|
coverPageUrl: book.coverPageUrl,
|
||||||
|
|
||||||
|
showCheckBuc: !this.showArchive && utils.hasProp(book, 'downloadSize') && book.url.indexOf('disk://') !== 0,
|
||||||
|
checkBuc: book.checkBuc,
|
||||||
|
needBookUpdate: (
|
||||||
|
!this.showArchive
|
||||||
|
&& book.checkBuc
|
||||||
|
&& book.bucSize
|
||||||
|
&& utils.hasProp(book, 'downloadSize')
|
||||||
|
&& book.bucSize !== book.downloadSize
|
||||||
|
&& (book.bucSize - book.downloadSize >= this.bucSizeDiff)
|
||||||
|
),
|
||||||
|
bucSize: book.bucSize,
|
||||||
|
downloadSize: book.downloadSize,
|
||||||
|
|
||||||
//для сортировки
|
//для сортировки
|
||||||
loadTimeRaw,
|
loadTimeRaw,
|
||||||
@@ -299,23 +430,25 @@ class RecentBooksPage {
|
|||||||
}
|
}
|
||||||
|
|
||||||
//нумерация
|
//нумерация
|
||||||
let num = 0;
|
|
||||||
|
|
||||||
result.sort((a, b) => b.loadTimeRaw - a.loadTimeRaw);
|
result.sort((a, b) => b.loadTimeRaw - a.loadTimeRaw);
|
||||||
for (const book of result) {
|
let num = 0;
|
||||||
|
for (let i = result.length - 1; i >= 0; i--) {
|
||||||
num++;
|
num++;
|
||||||
book.num = num;
|
result[i].num = num;
|
||||||
}
|
}
|
||||||
|
|
||||||
//фильтрация
|
//фильтрация
|
||||||
const search = this.search;
|
const search = this.search;
|
||||||
if (search) {
|
if (search) {
|
||||||
|
const lowerSearch = search.toLowerCase();
|
||||||
|
|
||||||
result = result.filter(item => {
|
result = result.filter(item => {
|
||||||
return !search ||
|
return !search
|
||||||
item.touchTime.includes(search) ||
|
|| item.touchTime.includes(search)
|
||||||
item.loadTime.includes(search) ||
|
|| item.loadTime.includes(search)
|
||||||
item.desc.title.toLowerCase().includes(search.toLowerCase()) ||
|
|| item.desc.title.toLowerCase().includes(lowerSearch)
|
||||||
item.desc.author.toLowerCase().includes(search.toLowerCase())
|
|| item.desc.author.toLowerCase().includes(lowerSearch)
|
||||||
|
;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -348,6 +481,7 @@ class RecentBooksPage {
|
|||||||
}
|
}
|
||||||
|
|
||||||
//группировка
|
//группировка
|
||||||
|
let nbuCount = 0;
|
||||||
const groups = {};
|
const groups = {};
|
||||||
const parents = {};
|
const parents = {};
|
||||||
let newResult = [];
|
let newResult = [];
|
||||||
@@ -364,13 +498,20 @@ class RecentBooksPage {
|
|||||||
if (book.active)
|
if (book.active)
|
||||||
parents[book.sameBookKey].activeParent = true;
|
parents[book.sameBookKey].activeParent = true;
|
||||||
|
|
||||||
|
book.showCheckBuc = false;
|
||||||
|
book.needBookUpdate = false;
|
||||||
|
|
||||||
groups[book.sameBookKey].push(book);
|
groups[book.sameBookKey].push(book);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
newResult.push(book);
|
newResult.push(book);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (book.needBookUpdate)
|
||||||
|
nbuCount++;
|
||||||
}
|
}
|
||||||
result = newResult;
|
result = newResult;
|
||||||
|
this.needBookUpdateCount = nbuCount;
|
||||||
|
|
||||||
//showSameBook
|
//showSameBook
|
||||||
if (this.showSameBook) {
|
if (this.showSameBook) {
|
||||||
@@ -387,6 +528,11 @@ class RecentBooksPage {
|
|||||||
result = newResult;
|
result = newResult;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//showNeedBookUpdateOnly
|
||||||
|
if (this.showNeedBookUpdateOnly) {
|
||||||
|
result = result.filter(item => item.needBookUpdate);
|
||||||
|
}
|
||||||
|
|
||||||
//другие стадии
|
//другие стадии
|
||||||
//.....
|
//.....
|
||||||
|
|
||||||
@@ -404,7 +550,9 @@ class RecentBooksPage {
|
|||||||
wordEnding(num, type = 0) {
|
wordEnding(num, type = 0) {
|
||||||
const endings = [
|
const endings = [
|
||||||
['ов', '', 'а', 'а', 'а', 'ов', 'ов', 'ов', 'ов', 'ов'],
|
['ов', '', 'а', 'а', 'а', 'ов', 'ов', 'ов', 'ов', 'ов'],
|
||||||
['й', 'я', 'и', 'и', 'и', 'й', 'й', 'й', 'й', 'й']
|
['й', 'я', 'и', 'и', 'и', 'й', 'й', 'й', 'й', 'й'],
|
||||||
|
['о', '', 'о', 'о', 'о', 'о', 'о', 'о', 'о', 'о'],
|
||||||
|
['ий', 'ие', 'ия', 'ия', 'ия', 'ий', 'ий', 'ий', 'ий', 'ий']
|
||||||
];
|
];
|
||||||
const deci = num % 100;
|
const deci = num % 100;
|
||||||
if (deci > 10 && deci < 20) {
|
if (deci > 10 && deci < 20) {
|
||||||
@@ -416,7 +564,7 @@ class RecentBooksPage {
|
|||||||
|
|
||||||
get header() {
|
get header() {
|
||||||
const len = (this.tableData ? this.tableData.length : 0);
|
const len = (this.tableData ? this.tableData.length : 0);
|
||||||
return `${(this.search ? 'Найдено' : 'Всего')} ${len} файл${this.wordEnding(len)}`;
|
return `${(this.search || this.showNeedBookUpdateOnly ? `Найден${this.wordEnding(len, 2)}` : 'Всего')} ${len} файл${this.wordEnding(len)}${this.showArchive ? ' в архиве' : ''}`;
|
||||||
}
|
}
|
||||||
|
|
||||||
async downloadBook(fb2path, fullTitle) {
|
async downloadBook(fb2path, fullTitle) {
|
||||||
@@ -442,15 +590,27 @@ class RecentBooksPage {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async handleDel(key) {
|
async handleDel(key) {
|
||||||
await bookManager.delRecentBook({key});
|
if (!this.showArchive) {
|
||||||
//this.updateTableData();//обновление уже происходит Reader.bookManagerEvent
|
await bookManager.delRecentBook({key});
|
||||||
|
this.$root.notify.info('Перенесено в архив');
|
||||||
if (!bookManager.mostRecentBook())
|
} else {
|
||||||
this.close();
|
if (await this.$root.stdDialog.confirm('Подтвердите удаление из архива:', ' ')) {
|
||||||
|
await bookManager.delRecentBook({key}, 2);
|
||||||
|
this.$root.notify.info('Удалено безвозвратно');
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
loadBook(row) {
|
async handleRestore(key) {
|
||||||
this.$emit('load-book', {url: row.url, path: row.path});
|
await bookManager.restoreRecentBook({key});
|
||||||
|
this.$root.notify.info('Восстановлено из архива');
|
||||||
|
}
|
||||||
|
|
||||||
|
async loadBook(item, force = false) {
|
||||||
|
if (item.deleted)
|
||||||
|
await this.handleRestore(item.key);
|
||||||
|
|
||||||
|
this.$emit('load-book', {url: item.url, path: item.path, force});
|
||||||
this.close();
|
this.close();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -507,6 +667,8 @@ class RecentBooksPage {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async scrollToActiveBook() {
|
async scrollToActiveBook() {
|
||||||
|
await this.$nextTick();
|
||||||
|
|
||||||
this.lockScroll = true;
|
this.lockScroll = true;
|
||||||
try {
|
try {
|
||||||
let activeIndex = -1;
|
let activeIndex = -1;
|
||||||
@@ -552,6 +714,16 @@ class RecentBooksPage {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async scrollRefresh() {
|
||||||
|
this.lockScroll = true;
|
||||||
|
await utils.sleep(100);
|
||||||
|
try {
|
||||||
|
this.$refs.virtualScroll.refresh();
|
||||||
|
} finally {
|
||||||
|
await utils.sleep(100);
|
||||||
|
this.lockScroll = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
get sortMethodOptions() {
|
get sortMethodOptions() {
|
||||||
return [
|
return [
|
||||||
@@ -566,6 +738,13 @@ class RecentBooksPage {
|
|||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
showArchiveToggle() {
|
||||||
|
this.showArchive = !this.showArchive;
|
||||||
|
this.showNeedBookUpdateOnly = false;
|
||||||
|
|
||||||
|
this.updateTableData();
|
||||||
|
}
|
||||||
|
|
||||||
close() {
|
close() {
|
||||||
this.$emit('recent-books-close');
|
this.$emit('recent-books-close');
|
||||||
}
|
}
|
||||||
@@ -576,6 +755,80 @@ class RecentBooksPage {
|
|||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
makeCoverHtml(data) {
|
||||||
|
return `<img src="${data}" style="height: 100%; width: 100%; object-fit: contain" />`;
|
||||||
|
}
|
||||||
|
|
||||||
|
isLoadedCover(coverPageUrl) {
|
||||||
|
if (!coverPageUrl)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
let loadedCover = this.covers[coverPageUrl];
|
||||||
|
|
||||||
|
if (loadedCover == 'error')
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if (!loadedCover) {
|
||||||
|
(async() => {
|
||||||
|
if (this.coversLoadFunc[coverPageUrl])
|
||||||
|
return;
|
||||||
|
|
||||||
|
this.coversLoadFunc[coverPageUrl] = (async() => {
|
||||||
|
//сначала заглянем в storage
|
||||||
|
let data = await coversStorage.getData(coverPageUrl);
|
||||||
|
if (data) {
|
||||||
|
this.covers[coverPageUrl] = this.makeCoverHtml(data);
|
||||||
|
} else {//иначе идем на сервер
|
||||||
|
try {
|
||||||
|
data = await readerApi.getUploadedFileBuf(coverPageUrl);
|
||||||
|
await coversStorage.setData(coverPageUrl, data);
|
||||||
|
this.covers[coverPageUrl] = this.makeCoverHtml(data);
|
||||||
|
} catch (e) {
|
||||||
|
console.error(e);
|
||||||
|
this.covers[coverPageUrl] = 'error';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
try {
|
||||||
|
await this.coversLoadFunc[coverPageUrl]();
|
||||||
|
} finally {
|
||||||
|
this.coversLoadFunc[coverPageUrl] = null;
|
||||||
|
}
|
||||||
|
})();
|
||||||
|
}
|
||||||
|
|
||||||
|
return (loadedCover != undefined);
|
||||||
|
}
|
||||||
|
|
||||||
|
getCoverHtml(coverPageUrl) {
|
||||||
|
if (coverPageUrl && this.covers[coverPageUrl])
|
||||||
|
return this.covers[coverPageUrl];
|
||||||
|
else
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
|
||||||
|
async checkBucChange(item) {
|
||||||
|
const book = await bookManager.getRecentBook(item);
|
||||||
|
if (book) {
|
||||||
|
await bookManager.setCheckBuc(book, item.checkBuc);
|
||||||
|
|
||||||
|
this.$root.notify.info(item.checkBuc
|
||||||
|
? 'Проверка обновлений книги включена'
|
||||||
|
: 'Проверка обновлений книги отключена'
|
||||||
|
);
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
showNeedBookUpdateOnlyToggle() {
|
||||||
|
this.showNeedBookUpdateOnly = !this.showNeedBookUpdateOnly;
|
||||||
|
this.showArchive = false;
|
||||||
|
|
||||||
|
this.updateTableData();
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export default vueComponent(RecentBooksPage);
|
export default vueComponent(RecentBooksPage);
|
||||||
@@ -600,11 +853,10 @@ export default vueComponent(RecentBooksPage);
|
|||||||
|
|
||||||
.table-row {
|
.table-row {
|
||||||
min-height: 80px;
|
min-height: 80px;
|
||||||
border-bottom: 1px solid #cccccc;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.row-part {
|
.row-part {
|
||||||
padding: 4px 4px 4px 4px;
|
padding: 4px 0px 4px 0px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.clickable {
|
.clickable {
|
||||||
@@ -612,18 +864,11 @@ export default vueComponent(RecentBooksPage);
|
|||||||
}
|
}
|
||||||
|
|
||||||
.break-word {
|
.break-word {
|
||||||
line-height: 180%;
|
|
||||||
overflow-wrap: break-word;
|
overflow-wrap: break-word;
|
||||||
word-wrap: break-word;
|
word-wrap: break-word;
|
||||||
white-space: normal;
|
white-space: normal;
|
||||||
}
|
}
|
||||||
|
|
||||||
.read-bar {
|
|
||||||
height: 3px;
|
|
||||||
background-color: #aaaaaa;
|
|
||||||
margin-bottom: 2px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.even {
|
.even {
|
||||||
background-color: #f2f2f2;
|
background-color: #f2f2f2;
|
||||||
}
|
}
|
||||||
@@ -644,18 +889,6 @@ export default vueComponent(RecentBooksPage);
|
|||||||
transform: rotate(90deg);
|
transform: rotate(90deg);
|
||||||
}
|
}
|
||||||
|
|
||||||
.row-info-top {
|
|
||||||
line-height: 110%;
|
|
||||||
border-left: 1px solid #cccccc;
|
|
||||||
border-bottom: 1px solid #cccccc;
|
|
||||||
}
|
|
||||||
|
|
||||||
.row-info-bottom {
|
|
||||||
line-height: 110%;
|
|
||||||
border: 1px solid #cccccc;
|
|
||||||
border-right: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.tool-button {
|
.tool-button {
|
||||||
min-width: 30px;
|
min-width: 30px;
|
||||||
width: 30px;
|
width: 30px;
|
||||||
@@ -664,4 +897,85 @@ export default vueComponent(RecentBooksPage);
|
|||||||
margin: 10px 6px 0px 3px;
|
margin: 10px 6px 0px 3px;
|
||||||
background-color: white;
|
background-color: white;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.row-info-bottom {
|
||||||
|
line-height: 110%;
|
||||||
|
border-left: 1px solid #cccccc;
|
||||||
|
border-bottom: 1px solid #cccccc;
|
||||||
|
height: 14px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.row-info-top {
|
||||||
|
line-height: 110%;
|
||||||
|
border: 1px solid #cccccc;
|
||||||
|
border-right: 0;
|
||||||
|
height: 14px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.time-info, .row-info-top {
|
||||||
|
color: #888888;
|
||||||
|
}
|
||||||
|
|
||||||
|
.read-bar {
|
||||||
|
height: 6px;
|
||||||
|
background-color: #b8b8b8;
|
||||||
|
}
|
||||||
|
|
||||||
|
.del-button {
|
||||||
|
width: 25px;
|
||||||
|
height: 20px;
|
||||||
|
position: absolute;
|
||||||
|
border-left: 1px solid #cccccc;
|
||||||
|
border-bottom: 1px solid #cccccc;
|
||||||
|
border-radius: 0 0 0 10px;
|
||||||
|
margin: 1px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.del-button:hover {
|
||||||
|
color: white;
|
||||||
|
background-color: #FF3030;
|
||||||
|
}
|
||||||
|
|
||||||
|
.restore-button {
|
||||||
|
width: 25px;
|
||||||
|
height: 20px;
|
||||||
|
position: absolute;
|
||||||
|
border-right: 1px solid #cccccc;
|
||||||
|
border-bottom: 1px solid #cccccc;
|
||||||
|
border-radius: 0 0 10px 0;
|
||||||
|
margin: 1px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.restore-button:hover {
|
||||||
|
color: white;
|
||||||
|
background-color: #00bb00;
|
||||||
|
}
|
||||||
|
|
||||||
|
.header-button, .header-button-pressed {
|
||||||
|
width: 80px;
|
||||||
|
height: 30px;
|
||||||
|
cursor: pointer;
|
||||||
|
color: #555555;
|
||||||
|
}
|
||||||
|
|
||||||
|
.header-button-update, .header-button-update-pressed {
|
||||||
|
width: 120px;
|
||||||
|
height: 30px;
|
||||||
|
cursor: pointer;
|
||||||
|
color: white;
|
||||||
|
}
|
||||||
|
|
||||||
|
.header-button:hover, .header-button-update:hover {
|
||||||
|
color: white;
|
||||||
|
background-color: #39902F;
|
||||||
|
}
|
||||||
|
|
||||||
|
.header-button-pressed, .header-button-update-pressed {
|
||||||
|
color: black;
|
||||||
|
background-color: yellow;
|
||||||
|
}
|
||||||
|
|
||||||
|
.buc-checkbox {
|
||||||
|
position: absolute;
|
||||||
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -12,6 +12,7 @@ import bookManager from '../share/bookManager';
|
|||||||
import readerApi from '../../../api/reader';
|
import readerApi from '../../../api/reader';
|
||||||
import * as utils from '../../../share/utils';
|
import * as utils from '../../../share/utils';
|
||||||
import * as cryptoUtils from '../../../share/cryptoUtils';
|
import * as cryptoUtils from '../../../share/cryptoUtils';
|
||||||
|
import LockQueue from '../../../share/LockQueue';
|
||||||
|
|
||||||
import localForage from 'localforage';
|
import localForage from 'localforage';
|
||||||
const ssCacheStore = localForage.createInstance({
|
const ssCacheStore = localForage.createInstance({
|
||||||
@@ -48,6 +49,8 @@ class ServerStorage {
|
|||||||
this.keyInited = false;
|
this.keyInited = false;
|
||||||
this.commit = this.$store.commit;
|
this.commit = this.$store.commit;
|
||||||
this.prevServerStorageKey = null;
|
this.prevServerStorageKey = null;
|
||||||
|
this.lock = new LockQueue(100);
|
||||||
|
|
||||||
this.$root.generateNewServerStorageKey = () => {this.generateNewServerStorageKey()};
|
this.$root.generateNewServerStorageKey = () => {this.generateNewServerStorageKey()};
|
||||||
|
|
||||||
this.debouncedSaveSettings = _.debounce(() => {
|
this.debouncedSaveSettings = _.debounce(() => {
|
||||||
@@ -542,14 +545,16 @@ class ServerStorage {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
async saveRecent(itemKey, recurse) {
|
async saveRecent(itemKeys, recurse) {
|
||||||
while (!this.inited || this.savingRecent)
|
while (!this.inited)
|
||||||
await utils.sleep(100);
|
await utils.sleep(100);
|
||||||
|
|
||||||
if (!this.keyInited || !this.serverSyncEnabled || this.savingRecent)
|
if (!this.keyInited || !this.serverSyncEnabled)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
this.savingRecent = true;
|
let needRecurseCall = false;
|
||||||
|
|
||||||
|
await this.lock.get();
|
||||||
try {
|
try {
|
||||||
const bm = bookManager;
|
const bm = bookManager;
|
||||||
|
|
||||||
@@ -559,22 +564,29 @@ class ServerStorage {
|
|||||||
|
|
||||||
//newRecentMod
|
//newRecentMod
|
||||||
let newRecentMod = {};
|
let newRecentMod = {};
|
||||||
if (itemKey && this.cachedRecentPatch.data[itemKey] && this.prevItemKey == itemKey) {
|
let oneItemKey = null;
|
||||||
|
if (itemKeys && itemKeys.length == 1)
|
||||||
|
oneItemKey = itemKeys[0];
|
||||||
|
|
||||||
|
if (oneItemKey && this.cachedRecentPatch.data[oneItemKey] && this.prevItemKey == oneItemKey) {
|
||||||
newRecentMod = _.cloneDeep(this.cachedRecentMod);
|
newRecentMod = _.cloneDeep(this.cachedRecentMod);
|
||||||
newRecentMod.rev++;
|
newRecentMod.rev++;
|
||||||
|
|
||||||
newRecentMod.data.key = itemKey;
|
newRecentMod.data.key = oneItemKey;
|
||||||
newRecentMod.data.mod = utils.getObjDiff(this.cachedRecentPatch.data[itemKey], bm.recent[itemKey]);
|
newRecentMod.data.mod = utils.getObjDiff(this.cachedRecentPatch.data[oneItemKey], bm.recent[oneItemKey]);
|
||||||
needSaveRecentMod = true;
|
needSaveRecentMod = true;
|
||||||
}
|
}
|
||||||
this.prevItemKey = itemKey;
|
this.prevItemKey = oneItemKey;
|
||||||
|
|
||||||
//newRecentPatch
|
//newRecentPatch
|
||||||
let newRecentPatch = {};
|
let newRecentPatch = {};
|
||||||
if (itemKey && !needSaveRecentMod) {
|
if (itemKeys && !needSaveRecentMod) {
|
||||||
newRecentPatch = _.cloneDeep(this.cachedRecentPatch);
|
newRecentPatch = _.cloneDeep(this.cachedRecentPatch);
|
||||||
newRecentPatch.rev++;
|
newRecentPatch.rev++;
|
||||||
newRecentPatch.data[itemKey] = _.cloneDeep(bm.recent[itemKey]);
|
|
||||||
|
for (const key of itemKeys) {
|
||||||
|
newRecentPatch.data[key] = _.cloneDeep(bm.recent[key]);
|
||||||
|
}
|
||||||
|
|
||||||
const applyMod = this.cachedRecentMod.data;
|
const applyMod = this.cachedRecentMod.data;
|
||||||
if (applyMod && applyMod.key && newRecentPatch.data[applyMod.key])
|
if (applyMod && applyMod.key && newRecentPatch.data[applyMod.key])
|
||||||
@@ -587,11 +599,7 @@ class ServerStorage {
|
|||||||
|
|
||||||
//newRecent
|
//newRecent
|
||||||
let newRecent = {};
|
let newRecent = {};
|
||||||
if (!itemKey || (needSaveRecentPatch && Object.keys(newRecentPatch.data).length > 10)) {
|
if (!itemKeys || (needSaveRecentPatch && Object.keys(newRecentPatch.data).length > 10)) {
|
||||||
//ждем весь bm.recent
|
|
||||||
/*while (!bookManager.loaded)
|
|
||||||
await utils.sleep(100);*/
|
|
||||||
|
|
||||||
newRecent = {rev: this.cachedRecent.rev + 1, data: _.cloneDeep(bm.recent)};
|
newRecent = {rev: this.cachedRecent.rev + 1, data: _.cloneDeep(bm.recent)};
|
||||||
newRecentPatch = {rev: this.cachedRecentPatch.rev + 1, data: {}};
|
newRecentPatch = {rev: this.cachedRecentPatch.rev + 1, data: {}};
|
||||||
newRecentMod = {rev: this.cachedRecentMod.rev + 1, data: {}};
|
newRecentMod = {rev: this.cachedRecentMod.rev + 1, data: {}};
|
||||||
@@ -625,10 +633,8 @@ class ServerStorage {
|
|||||||
|
|
||||||
if (res)
|
if (res)
|
||||||
this.warning(`Последние изменения отменены. Данные синхронизированы с сервером.`);
|
this.warning(`Последние изменения отменены. Данные синхронизированы с сервером.`);
|
||||||
if (!recurse && itemKey) {
|
if (!recurse && itemKeys) {
|
||||||
this.savingRecent = false;
|
needRecurseCall = true;
|
||||||
await this.saveRecent(itemKey, true);
|
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
} else if (result.state == 'success') {
|
} else if (result.state == 'success') {
|
||||||
if (needSaveRecent && newRecent.rev)
|
if (needSaveRecent && newRecent.rev)
|
||||||
@@ -639,8 +645,11 @@ class ServerStorage {
|
|||||||
await this.setCachedRecentMod(newRecentMod);
|
await this.setCachedRecentMod(newRecentMod);
|
||||||
}
|
}
|
||||||
} finally {
|
} finally {
|
||||||
this.savingRecent = false;
|
this.lock.ret();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (needRecurseCall)
|
||||||
|
await this.saveRecent(itemKeys, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
async storageCheck(items) {
|
async storageCheck(items) {
|
||||||
|
|||||||
@@ -1,19 +1,21 @@
|
|||||||
<template>
|
<template>
|
||||||
<Window ref="window" height="140px" max-width="600px" :top-shift="-50" @close="close">
|
<Window ref="window" height="125px" max-width="600px" :top-shift="-50" @close="close">
|
||||||
<template #header>
|
<template #header>
|
||||||
Установить позицию
|
Установить позицию
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<div id="set-position-slider" class="slider q-px-md">
|
<div class="col column justify-center">
|
||||||
<q-slider
|
<div id="set-position-slider" class="slider q-px-md column justify-center">
|
||||||
v-model="sliderValue"
|
<q-slider
|
||||||
thumb-path="M 2, 10 a 8.5,8.5 0 1,0 17,0 a 8.5,8.5 0 1,0 -17,0"
|
v-model="sliderValue"
|
||||||
|
thumb-path="M 2, 10 a 8.5,8.5 0 1,0 17,0 a 8.5,8.5 0 1,0 -17,0"
|
||||||
:max="sliderMax"
|
|
||||||
label
|
:max="sliderMax"
|
||||||
:label-value="(sliderMax ? (sliderValue/sliderMax*100).toFixed(2) + '%' : 0)"
|
label
|
||||||
color="primary"
|
:label-value="(sliderMax ? (sliderValue/sliderMax*100).toFixed(2) + '%' : 0)"
|
||||||
/>
|
color="primary"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</Window>
|
</Window>
|
||||||
</template>
|
</template>
|
||||||
@@ -76,7 +78,8 @@ export default vueComponent(SetPositionPage);
|
|||||||
|
|
||||||
<style scoped>
|
<style scoped>
|
||||||
.slider {
|
.slider {
|
||||||
margin: 20px;
|
margin: 0 20px 0 20px;
|
||||||
|
height: 35px;
|
||||||
background-color: #efefef;
|
background-color: #efefef;
|
||||||
border-radius: 15px;
|
border-radius: 15px;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -43,25 +43,14 @@
|
|||||||
|
|
||||||
<div class="item row">
|
<div class="item row">
|
||||||
<div class="label-6">Уведомление</div>
|
<div class="label-6">Уведомление</div>
|
||||||
<q-checkbox size="xs" v-model="showNeedUpdateNotify">
|
<q-checkbox size="xs" v-model="showDonationDialog">
|
||||||
Показывать уведомление о новой версии
|
Показывать форму доната
|
||||||
<q-tooltip :delay="1000" anchor="top middle" self="bottom middle" content-style="font-size: 80%">
|
<q-tooltip :delay="1000" anchor="top middle" self="bottom middle" content-style="font-size: 80%">
|
||||||
Напоминать о необходимости обновления страницы<br>
|
Показывать диалог для сбора пожертвований
|
||||||
при появлении новой версии читалки
|
|
||||||
</q-tooltip>
|
</q-tooltip>
|
||||||
</q-checkbox>
|
</q-checkbox>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<!--div class="item row">
|
|
||||||
<div class="label-6">Уведомление</div>
|
|
||||||
<q-checkbox size="xs" v-model="showDonationDialog2020">
|
|
||||||
Показывать "Оплатим хостинг вместе"
|
|
||||||
<q-tooltip :delay="1000" anchor="top middle" self="bottom middle" content-style="font-size: 80%">
|
|
||||||
Показывать уведомление "Оплатим хостинг вместе"
|
|
||||||
</q-tooltip>
|
|
||||||
</q-checkbox>
|
|
||||||
</div-->
|
|
||||||
|
|
||||||
<!---------------------------------------------->
|
<!---------------------------------------------->
|
||||||
<div class="part-header">Другое</div>
|
<div class="part-header">Другое</div>
|
||||||
|
|
||||||
|
|||||||
@@ -5,6 +5,8 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<div class="col row">
|
<div class="col row">
|
||||||
|
<a ref="download" style="display: none;" target="_blank"></a>
|
||||||
|
|
||||||
<div class="full-height">
|
<div class="full-height">
|
||||||
<q-tabs
|
<q-tabs
|
||||||
ref="tabs"
|
ref="tabs"
|
||||||
@@ -28,6 +30,7 @@
|
|||||||
<q-tab class="tab" name="keys" icon="la la-gamepad" label="Управление" />
|
<q-tab class="tab" name="keys" icon="la la-gamepad" label="Управление" />
|
||||||
<q-tab class="tab" name="pagemove" icon="la la-school" label="Листание" />
|
<q-tab class="tab" name="pagemove" icon="la la-school" label="Листание" />
|
||||||
<q-tab class="tab" name="convert" icon="la la-magic" label="Конвертир." />
|
<q-tab class="tab" name="convert" icon="la la-magic" label="Конвертир." />
|
||||||
|
<q-tab class="tab" name="update" icon="la la-sync" label="Обновление" />
|
||||||
<q-tab class="tab" name="others" icon="la la-list-ul" label="Прочее" />
|
<q-tab class="tab" name="others" icon="la la-list-ul" label="Прочее" />
|
||||||
<q-tab class="tab" name="reset" icon="la la-broom" label="Сброс" />
|
<q-tab class="tab" name="reset" icon="la la-broom" label="Сброс" />
|
||||||
<div v-show="tabsScrollable" class="q-pt-lg" />
|
<div v-show="tabsScrollable" class="q-pt-lg" />
|
||||||
@@ -97,6 +100,10 @@
|
|||||||
<div v-if="selectedTab == 'convert'" class="fit tab-panel">
|
<div v-if="selectedTab == 'convert'" class="fit tab-panel">
|
||||||
@@include('./ConvertTab.inc');
|
@@include('./ConvertTab.inc');
|
||||||
</div>
|
</div>
|
||||||
|
<!-- Обновление ------------------------------------------------------------------>
|
||||||
|
<div v-if="selectedTab == 'update'" class="fit tab-panel">
|
||||||
|
@@include('./UpdateTab.inc');
|
||||||
|
</div>
|
||||||
<!-- Прочее ---------------------------------------------------------------------->
|
<!-- Прочее ---------------------------------------------------------------------->
|
||||||
<div v-if="selectedTab == 'others'" class="fit tab-panel">
|
<div v-if="selectedTab == 'others'" class="fit tab-panel">
|
||||||
@@include('./OthersTab.inc');
|
@@include('./OthersTab.inc');
|
||||||
@@ -124,6 +131,7 @@ import NumInput from '../../share/NumInput.vue';
|
|||||||
import UserHotKeys from './UserHotKeys/UserHotKeys.vue';
|
import UserHotKeys from './UserHotKeys/UserHotKeys.vue';
|
||||||
import wallpaperStorage from '../share/wallpaperStorage';
|
import wallpaperStorage from '../share/wallpaperStorage';
|
||||||
|
|
||||||
|
import readerApi from '../../../api/reader';
|
||||||
import rstore from '../../../store/modules/reader';
|
import rstore from '../../../store/modules/reader';
|
||||||
import defPalette from './defPalette';
|
import defPalette from './defPalette';
|
||||||
|
|
||||||
@@ -310,6 +318,10 @@ class SettingsPage {
|
|||||||
return this.$store.state.reader.profiles;
|
return this.$store.state.reader.profiles;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
get configBucEnabled() {
|
||||||
|
return this.$store.state.config.bucEnabled;
|
||||||
|
}
|
||||||
|
|
||||||
get currentProfileOptions() {
|
get currentProfileOptions() {
|
||||||
const profNames = Object.keys(this.profiles)
|
const profNames = Object.keys(this.profiles)
|
||||||
profNames.sort();
|
profNames.sort();
|
||||||
@@ -636,8 +648,17 @@ class SettingsPage {
|
|||||||
|
|
||||||
if (index < 0)
|
if (index < 0)
|
||||||
newUserWallpapers.push({label, cssClass});
|
newUserWallpapers.push({label, cssClass});
|
||||||
if (!wallpaperStorage.keyExists(cssClass))
|
if (!wallpaperStorage.keyExists(cssClass)) {
|
||||||
await wallpaperStorage.setData(cssClass, data);
|
await wallpaperStorage.setData(cssClass, data);
|
||||||
|
//отправим data на сервер в файл `/upload/${key}`
|
||||||
|
try {
|
||||||
|
//const res =
|
||||||
|
await readerApi.uploadFileBuf(data);
|
||||||
|
//console.log(res);
|
||||||
|
} catch (e) {
|
||||||
|
console.error(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
this.userWallpapers = newUserWallpapers;
|
this.userWallpapers = newUserWallpapers;
|
||||||
this.wallpaper = cssClass;
|
this.wallpaper = cssClass;
|
||||||
@@ -664,6 +685,27 @@ class SettingsPage {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async downloadWallpaper() {
|
||||||
|
if (this.wallpaper.indexOf('user-paper') != 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
try {
|
||||||
|
const d = this.$refs.download;
|
||||||
|
|
||||||
|
const dataUrl = await wallpaperStorage.getData(this.wallpaper);
|
||||||
|
|
||||||
|
if (!dataUrl)
|
||||||
|
throw new Error('Файл обоев не найден');
|
||||||
|
|
||||||
|
d.href = dataUrl;
|
||||||
|
d.download = `wallpaper-#${this.wallpaper.replace('user-paper', '').substring(0, 4)}`;
|
||||||
|
|
||||||
|
d.click();
|
||||||
|
} catch (e) {
|
||||||
|
this.$root.stdDialog.alert(e.message, 'Ошибка', {color: 'negative'});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
keyHook(event) {
|
keyHook(event) {
|
||||||
if (!this.$root.stdDialog.active && event.type == 'keydown' && event.key == 'Escape') {
|
if (!this.$root.stdDialog.active && event.type == 'keydown' && event.key == 'Escape') {
|
||||||
this.close();
|
this.close();
|
||||||
|
|||||||
76
client/components/Reader/SettingsPage/UpdateTab.inc
Normal file
76
client/components/Reader/SettingsPage/UpdateTab.inc
Normal file
@@ -0,0 +1,76 @@
|
|||||||
|
<!---------------------------------------------->
|
||||||
|
<div class="part-header">Обновление читалки</div>
|
||||||
|
<div class="item row">
|
||||||
|
<div class="label-6"></div>
|
||||||
|
<q-checkbox size="xs" v-model="showNeedUpdateNotify">
|
||||||
|
Проверять наличие новой версии
|
||||||
|
<q-tooltip :delay="1000" anchor="top middle" self="bottom middle" content-style="font-size: 80%">
|
||||||
|
Напоминать о необходимости обновления страницы<br>
|
||||||
|
при появлении новой версии читалки
|
||||||
|
</q-tooltip>
|
||||||
|
</q-checkbox>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!---------------------------------------------->
|
||||||
|
<div class="part-header">Обновление книг</div>
|
||||||
|
<div v-show="!configBucEnabled" class="item row">
|
||||||
|
<div class="label-6"></div>
|
||||||
|
<div>Сервер обновлений временно не работает</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div v-show="configBucEnabled" class="item row">
|
||||||
|
<div class="label-6"></div>
|
||||||
|
<q-checkbox size="xs" v-model="bucEnabled">
|
||||||
|
Проверять обновления книг
|
||||||
|
</q-checkbox>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div v-show="configBucEnabled && bucEnabled" class="item row">
|
||||||
|
<div class="label-6"></div>
|
||||||
|
<div class="col-5 column justify-center items-end q-pr-xs">Разница размеров</div>
|
||||||
|
<div class="col row">
|
||||||
|
<NumInput class="col-left" v-model="bucSizeDiff" />
|
||||||
|
|
||||||
|
<q-tooltip :delay="1000" anchor="top middle" self="bottom middle" content-style="font-size: 80%">
|
||||||
|
Уведомлять о наличии обновления книги в списке загруженных<br>
|
||||||
|
при указанной разнице в размерах старого и нового файлов.<br>
|
||||||
|
Разница указывается в байтах и может быть отрицательной.
|
||||||
|
</q-tooltip>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div v-show="configBucEnabled && bucEnabled" class="item row">
|
||||||
|
<div class="label-6"></div>
|
||||||
|
<q-checkbox size="xs" v-model="bucSetOnNew">
|
||||||
|
Автопроверка для вновь загружаемых
|
||||||
|
<q-tooltip :delay="1000" anchor="top middle" self="bottom middle" content-style="font-size: 80%">
|
||||||
|
Автоматически устанавливать флаг проверки<br>
|
||||||
|
обновлений для всех вновь загружаемых книг
|
||||||
|
</q-tooltip>
|
||||||
|
</q-checkbox>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div v-show="configBucEnabled && bucEnabled" class="item row">
|
||||||
|
<div class="label-6"></div>
|
||||||
|
<q-checkbox size="xs" v-model="bucCancelEnabled">
|
||||||
|
Отменять проверку через {{ bucCancelDays }} дней{{ (bucCancelEnabled ? ':' : '') }}
|
||||||
|
<q-tooltip :delay="1000" anchor="top middle" self="bottom middle" content-style="font-size: 80%">
|
||||||
|
Снимать флаг проверки с книги, если не было<br>
|
||||||
|
обновлений в течение {{ bucCancelDays }} дней
|
||||||
|
</q-tooltip>
|
||||||
|
</q-checkbox>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div v-show="configBucEnabled && bucEnabled && bucCancelEnabled" class="item row">
|
||||||
|
<div class="label-6"></div>
|
||||||
|
<div class="col-5"></div>
|
||||||
|
<div class="col row">
|
||||||
|
<NumInput class="col-left" v-model="bucCancelDays" :min="1" :max="10000"/>
|
||||||
|
|
||||||
|
<q-tooltip :delay="1000" anchor="top middle" self="bottom middle" content-style="font-size: 80%">
|
||||||
|
Снимать флаг проверки с книги, если не было<br>
|
||||||
|
обновлений в течение {{ bucCancelDays }} дней
|
||||||
|
</q-tooltip>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
@@ -102,6 +102,11 @@
|
|||||||
Удалить выбранные обои
|
Удалить выбранные обои
|
||||||
</q-tooltip>
|
</q-tooltip>
|
||||||
</q-btn>
|
</q-btn>
|
||||||
|
<q-btn v-show="wallpaper.indexOf('user-paper') === 0" class="q-ml-sm" round dense color="blue" icon="la la-file-download" @click.stop="downloadWallpaper">
|
||||||
|
<q-tooltip :delay="1500" anchor="bottom middle" content-style="font-size: 80%">
|
||||||
|
Скачать выбранные обои
|
||||||
|
</q-tooltip>
|
||||||
|
</q-btn>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
|||||||
@@ -3,6 +3,8 @@ import sax from '../../../../server/core/sax';
|
|||||||
import * as utils from '../../../share/utils';
|
import * as utils from '../../../share/utils';
|
||||||
|
|
||||||
const maxImageLineCount = 100;
|
const maxImageLineCount = 100;
|
||||||
|
const maxParaLength = 10000;
|
||||||
|
const maxParaTextLength = 10000;
|
||||||
|
|
||||||
// defaults
|
// defaults
|
||||||
const defaultSettings = {
|
const defaultSettings = {
|
||||||
@@ -83,6 +85,7 @@ export default class BookParser {
|
|||||||
let binaryId = '';
|
let binaryId = '';
|
||||||
let binaryType = '';
|
let binaryType = '';
|
||||||
let dimPromises = [];
|
let dimPromises = [];
|
||||||
|
this.coverPageId = '';
|
||||||
|
|
||||||
//оглавление
|
//оглавление
|
||||||
this.contents = [];
|
this.contents = [];
|
||||||
@@ -226,13 +229,26 @@ export default class BookParser {
|
|||||||
paraOffset += len;
|
paraOffset += len;
|
||||||
};
|
};
|
||||||
|
|
||||||
const growParagraph = (text, len) => {
|
const growParagraph = (text, len, textRaw) => {
|
||||||
|
//начальный параграф
|
||||||
if (paraIndex < 0) {
|
if (paraIndex < 0) {
|
||||||
newParagraph();
|
newParagraph();
|
||||||
growParagraph(text, len);
|
growParagraph(text, len);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//ограничение на размер куска текста в параграфе
|
||||||
|
if (textRaw && textRaw.length > maxParaTextLength) {
|
||||||
|
while (textRaw.length > 0) {
|
||||||
|
const textPart = textRaw.substring(0, maxParaTextLength);
|
||||||
|
textRaw = textRaw.substring(maxParaTextLength);
|
||||||
|
|
||||||
|
newParagraph();
|
||||||
|
growParagraph(textPart, textPart.length);
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if (inSubtitle) {
|
if (inSubtitle) {
|
||||||
curSubtitle.title += text;
|
curSubtitle.title += text;
|
||||||
} else if (inTitle) {
|
} else if (inTitle) {
|
||||||
@@ -240,6 +256,14 @@ export default class BookParser {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const p = para[paraIndex];
|
const p = para[paraIndex];
|
||||||
|
|
||||||
|
//ограничение на размер параграфа
|
||||||
|
if (p.length > maxParaLength) {
|
||||||
|
newParagraph();
|
||||||
|
growParagraph(text, len);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
p.length += len;
|
p.length += len;
|
||||||
p.text += text;
|
p.text += text;
|
||||||
paraOffset += len;
|
paraOffset += len;
|
||||||
@@ -266,7 +290,7 @@ export default class BookParser {
|
|||||||
const href = attrs.href.value;
|
const href = attrs.href.value;
|
||||||
const alt = (attrs.alt && attrs.alt.value ? attrs.alt.value : '');
|
const alt = (attrs.alt && attrs.alt.value ? attrs.alt.value : '');
|
||||||
const {id, local} = this.imageHrefToId(href);
|
const {id, local} = this.imageHrefToId(href);
|
||||||
if (href[0] == '#') {//local
|
if (local) {//local
|
||||||
imageNum++;
|
imageNum++;
|
||||||
|
|
||||||
if (inPara && !this.sets.showInlineImagesInCenter && !center)
|
if (inPara && !this.sets.showInlineImagesInCenter && !center)
|
||||||
@@ -278,6 +302,11 @@ export default class BookParser {
|
|||||||
|
|
||||||
if (inPara && this.sets.showInlineImagesInCenter)
|
if (inPara && this.sets.showInlineImagesInCenter)
|
||||||
newParagraph();
|
newParagraph();
|
||||||
|
|
||||||
|
//coverpage
|
||||||
|
if (path == '/fictionbook/description/title-info/coverpage/image') {
|
||||||
|
this.coverPageId = id;
|
||||||
|
}
|
||||||
} else {//external
|
} else {//external
|
||||||
imageNum++;
|
imageNum++;
|
||||||
|
|
||||||
@@ -536,7 +565,7 @@ export default class BookParser {
|
|||||||
tClose += (center ? '</center>' : '');
|
tClose += (center ? '</center>' : '');
|
||||||
|
|
||||||
if (text != ' ')
|
if (text != ' ')
|
||||||
growParagraph(`${tOpen}${text}${tClose}`, text.length);
|
growParagraph(`${tOpen}${text}${tClose}`, text.length, text);
|
||||||
else
|
else
|
||||||
growParagraph(' ', 1);
|
growParagraph(' ', 1);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,8 +2,10 @@ import localForage from 'localforage';
|
|||||||
import path from 'path-browserify';
|
import path from 'path-browserify';
|
||||||
import _ from 'lodash';
|
import _ from 'lodash';
|
||||||
|
|
||||||
import * as utils from '../../../share/utils';
|
|
||||||
import BookParser from './BookParser';
|
import BookParser from './BookParser';
|
||||||
|
import readerApi from '../../../api/reader';
|
||||||
|
import coversStorage from './coversStorage';
|
||||||
|
import * as utils from '../../../share/utils';
|
||||||
|
|
||||||
const maxDataSize = 500*1024*1024;//compressed bytes
|
const maxDataSize = 500*1024*1024;//compressed bytes
|
||||||
const maxRecentLength = 5000;
|
const maxRecentLength = 5000;
|
||||||
@@ -232,6 +234,10 @@ class BookManager {
|
|||||||
|
|
||||||
async addBook(newBook, callback) {
|
async addBook(newBook, callback) {
|
||||||
let meta = {url: newBook.url, path: newBook.path};
|
let meta = {url: newBook.url, path: newBook.path};
|
||||||
|
|
||||||
|
if (newBook.downloadSize !== undefined && newBook.downloadSize >= 0)
|
||||||
|
meta.downloadSize = newBook.downloadSize;
|
||||||
|
|
||||||
meta.key = this.keyFromPath(meta.path);
|
meta.key = this.keyFromPath(meta.path);
|
||||||
meta.addTime = Date.now();//время добавления в кеш
|
meta.addTime = Date.now();//время добавления в кеш
|
||||||
|
|
||||||
@@ -345,9 +351,38 @@ class BookManager {
|
|||||||
const parsed = new BookParser(this.settings);
|
const parsed = new BookParser(this.settings);
|
||||||
|
|
||||||
const parsedMeta = await parsed.parse(data, callback);
|
const parsedMeta = await parsed.parse(data, callback);
|
||||||
|
|
||||||
|
//cover page
|
||||||
|
let coverPageUrl = '';
|
||||||
|
if (parsed.coverPageId && parsed.binary[parsed.coverPageId]) {
|
||||||
|
const bin = parsed.binary[parsed.coverPageId];
|
||||||
|
let dataUrl = `data:${bin.type};base64,${bin.data}`;
|
||||||
|
try {
|
||||||
|
dataUrl = await utils.resizeImage(dataUrl, 160, 160, 0.94);
|
||||||
|
} catch (e) {
|
||||||
|
console.error(e);
|
||||||
|
}
|
||||||
|
|
||||||
|
coverPageUrl = readerApi.makeUrlFromBuf(dataUrl);
|
||||||
|
|
||||||
|
//далее асинхронно
|
||||||
|
(async() => {
|
||||||
|
//отправим dataUrl на сервер в /upload
|
||||||
|
try {
|
||||||
|
await readerApi.uploadFileBuf(dataUrl, coverPageUrl);
|
||||||
|
} catch (e) {
|
||||||
|
console.error(e);
|
||||||
|
}
|
||||||
|
|
||||||
|
//сохраним в storage
|
||||||
|
await coversStorage.setData(coverPageUrl, dataUrl);
|
||||||
|
})();
|
||||||
|
}
|
||||||
|
|
||||||
const result = Object.assign({}, meta, parsedMeta, {
|
const result = Object.assign({}, meta, parsedMeta, {
|
||||||
length: data.length,
|
length: data.length,
|
||||||
textLength: parsed.textLength,
|
textLength: parsed.textLength,
|
||||||
|
coverPageUrl,
|
||||||
parsed
|
parsed
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -433,9 +468,9 @@ class BookManager {
|
|||||||
return this.recent[value.key];
|
return this.recent[value.key];
|
||||||
}
|
}
|
||||||
|
|
||||||
async delRecentBook(value) {
|
async delRecentBook(value, delFlag = 1) {
|
||||||
const item = this.recent[value.key];
|
const item = this.recent[value.key];
|
||||||
item.deleted = 1;
|
item.deleted = delFlag;
|
||||||
|
|
||||||
if (this.recentLastKey == value.key) {
|
if (this.recentLastKey == value.key) {
|
||||||
await this.recentSetLastKey(null);
|
await this.recentSetLastKey(null);
|
||||||
@@ -445,6 +480,38 @@ class BookManager {
|
|||||||
this.emit('recent-deleted', value.key);
|
this.emit('recent-deleted', value.key);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async restoreRecentBook(value) {
|
||||||
|
const item = this.recent[value.key];
|
||||||
|
item.deleted = 0;
|
||||||
|
|
||||||
|
await this.recentSetItem(item);
|
||||||
|
}
|
||||||
|
|
||||||
|
async setCheckBuc(value, checkBuc) {
|
||||||
|
const item = this.recent[value.key];
|
||||||
|
|
||||||
|
const updateItems = [];
|
||||||
|
if (item) {
|
||||||
|
if (item.sameBookKey !== undefined) {
|
||||||
|
const sorted = this.getSortedRecent();
|
||||||
|
for (const book of sorted) {
|
||||||
|
if (!book.deleted && book.sameBookKey === item.sameBookKey)
|
||||||
|
updateItems.push(book);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
updateItems.push(item);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const now = Date.now();
|
||||||
|
for (const book of updateItems) {
|
||||||
|
book.checkBuc = checkBuc;
|
||||||
|
if (checkBuc)
|
||||||
|
book.checkBucTime = now;
|
||||||
|
await this.recentSetItem(book);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
async cleanRecentBooks() {
|
async cleanRecentBooks() {
|
||||||
const sorted = this.getSortedRecent();
|
const sorted = this.getSortedRecent();
|
||||||
|
|
||||||
|
|||||||
61
client/components/Reader/share/coversStorage.js
Normal file
61
client/components/Reader/share/coversStorage.js
Normal file
@@ -0,0 +1,61 @@
|
|||||||
|
import localForage from 'localforage';
|
||||||
|
//import _ from 'lodash';
|
||||||
|
import * as utils from '../../../share/utils';
|
||||||
|
|
||||||
|
const maxDataSize = 100*1024*1024;
|
||||||
|
|
||||||
|
const coversStore = localForage.createInstance({
|
||||||
|
name: 'coversStorage'
|
||||||
|
});
|
||||||
|
|
||||||
|
class CoversStorage {
|
||||||
|
constructor() {
|
||||||
|
}
|
||||||
|
|
||||||
|
async init() {
|
||||||
|
this.cleanCovers(); //no await
|
||||||
|
}
|
||||||
|
|
||||||
|
async setData(key, data) {
|
||||||
|
await coversStore.setItem(key, {addTime: Date.now(), data});
|
||||||
|
}
|
||||||
|
|
||||||
|
async getData(key) {
|
||||||
|
const item = await coversStore.getItem(key);
|
||||||
|
return (item ? item.data : undefined);
|
||||||
|
}
|
||||||
|
|
||||||
|
async removeData(key) {
|
||||||
|
await coversStore.removeItem(key);
|
||||||
|
}
|
||||||
|
|
||||||
|
async cleanCovers() {
|
||||||
|
await utils.sleep(10000);
|
||||||
|
|
||||||
|
while (1) {// eslint-disable-line no-constant-condition
|
||||||
|
let size = 0;
|
||||||
|
let min = Date.now();
|
||||||
|
let toDel = null;
|
||||||
|
for (const key of (await coversStore.keys())) {
|
||||||
|
const item = await coversStore.getItem(key);
|
||||||
|
|
||||||
|
size += item.data.length;
|
||||||
|
|
||||||
|
if (item.addTime < min) {
|
||||||
|
toDel = key;
|
||||||
|
min = item.addTime;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
if (size > maxDataSize && toDel) {
|
||||||
|
await this.removeData(toDel);
|
||||||
|
} else {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
export default new CoversStorage();
|
||||||
@@ -32,6 +32,10 @@ class WallpaperStorage {
|
|||||||
this.cachedKeys = await wpStore.keys();
|
this.cachedKeys = await wpStore.keys();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async getKeys() {
|
||||||
|
return await wpStore.keys();
|
||||||
|
}
|
||||||
|
|
||||||
keyExists(key) {//не асинхронная
|
keyExists(key) {//не асинхронная
|
||||||
return this.cachedKeys.includes(key);
|
return this.cachedKeys.includes(key);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,4 +1,50 @@
|
|||||||
export const versionHistory = [
|
export const versionHistory = [
|
||||||
|
{
|
||||||
|
version: '0.12.1',
|
||||||
|
releaseDate: '2022-09-01',
|
||||||
|
showUntil: '2022-08-30',
|
||||||
|
content:
|
||||||
|
`
|
||||||
|
<ul>
|
||||||
|
<li>добавлена форма для доната</li>
|
||||||
|
<li>исправления багов</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
`
|
||||||
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
version: '0.12.0',
|
||||||
|
releaseDate: '2022-07-27',
|
||||||
|
showUntil: '2022-08-03',
|
||||||
|
content:
|
||||||
|
`
|
||||||
|
<ul>
|
||||||
|
<li>запущен сервер проверки обновлений книг:</li>
|
||||||
|
<ul>
|
||||||
|
<li>проверка обновления той или иной книги настраивается в списке загруженных (чекбокс)</li>
|
||||||
|
<li>для того, чтобы чекбокс появился у ранее загруженной, необходимо принудительно обновить книгу</li>
|
||||||
|
<li>в настройках можно указать разницу размеров, при которой требуется делать уведомление</li>
|
||||||
|
</ul>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
`
|
||||||
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
version: '0.11.8',
|
||||||
|
releaseDate: '2022-07-14',
|
||||||
|
showUntil: '2022-07-13',
|
||||||
|
content:
|
||||||
|
`
|
||||||
|
<ul>
|
||||||
|
<li>добавлено отображение и синхронизация обложек в окне загруженных книг</li>
|
||||||
|
<li>добавлена синхронизация обоев</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
`
|
||||||
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
version: '0.11.7',
|
version: '0.11.7',
|
||||||
releaseDate: '2022-07-12',
|
releaseDate: '2022-07-12',
|
||||||
|
|||||||
@@ -45,6 +45,8 @@ export function formatDate(d, format) {
|
|||||||
`${d.getHours().toString().padStart(2, '0')}:${d.getMinutes().toString().padStart(2, '0')}`;
|
`${d.getHours().toString().padStart(2, '0')}:${d.getMinutes().toString().padStart(2, '0')}`;
|
||||||
case 'coDate':
|
case 'coDate':
|
||||||
return `${d.getFullYear()}-${(d.getMonth() + 1).toString().padStart(2, '0')}-${d.getDate().toString().padStart(2, '0')}`;
|
return `${d.getFullYear()}-${(d.getMonth() + 1).toString().padStart(2, '0')}-${d.getDate().toString().padStart(2, '0')}`;
|
||||||
|
case 'coMonth':
|
||||||
|
return `${(d.getMonth() + 1).toString().padStart(2, '0')}-${d.getDate().toString().padStart(2, '0')}`;
|
||||||
case 'noDate':
|
case 'noDate':
|
||||||
return `${d.getDate().toString().padStart(2, '0')}.${(d.getMonth() + 1).toString().padStart(2, '0')}.${d.getFullYear()}`;
|
return `${d.getDate().toString().padStart(2, '0')}.${(d.getMonth() + 1).toString().padStart(2, '0')}.${d.getFullYear()}`;
|
||||||
}
|
}
|
||||||
@@ -363,4 +365,54 @@ export function getBookTitle(fb2) {
|
|||||||
]).join(' - ');
|
]).join(' - ');
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function resizeImage(dataUrl, toWidth, toHeight, quality = 0.9) {
|
||||||
|
return new Promise ((resolve, reject) => { (async() => {
|
||||||
|
const img = new Image();
|
||||||
|
|
||||||
|
let resolved = false;
|
||||||
|
img.onload = () => {
|
||||||
|
try {
|
||||||
|
let width = img.width;
|
||||||
|
let height = img.height;
|
||||||
|
|
||||||
|
if (width > height) {
|
||||||
|
if (width > toWidth) {
|
||||||
|
height = height * (toWidth / width);
|
||||||
|
width = toWidth;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (height > toHeight) {
|
||||||
|
width = width * (toHeight / height);
|
||||||
|
height = toHeight;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const canvas = document.createElement('canvas');
|
||||||
|
canvas.width = width;
|
||||||
|
canvas.height = height;
|
||||||
|
const ctx = canvas.getContext('2d');
|
||||||
|
ctx.drawImage(img, 0, 0, width, height);
|
||||||
|
const result = canvas.toDataURL('image/jpeg', quality);
|
||||||
|
resolved = true;
|
||||||
|
resolve(result);
|
||||||
|
} catch (e) {
|
||||||
|
reject(e);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
img.onerror = reject;
|
||||||
|
|
||||||
|
img.src = dataUrl;
|
||||||
|
|
||||||
|
await sleep(1000);
|
||||||
|
if (!resolved)
|
||||||
|
reject('Не удалось изменить размер');
|
||||||
|
})().catch(reject); });
|
||||||
|
}
|
||||||
|
|
||||||
|
export function makeDonation() {
|
||||||
|
window.open('https://donatty.com/liberama', '_blank');
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
import { createStore } from 'vuex';
|
import { createStore } from 'vuex';
|
||||||
import createPersistedState from 'vuex-persistedstate';
|
//import createPersistedState from 'vuex-persistedstate';
|
||||||
|
import VuexPersistence from 'vuex-persist';
|
||||||
|
|
||||||
import root from './root.js';
|
import root from './root.js';
|
||||||
import uistate from './modules/uistate';
|
import uistate from './modules/uistate';
|
||||||
@@ -8,6 +9,8 @@ import reader from './modules/reader';
|
|||||||
|
|
||||||
const debug = process.env.NODE_ENV !== 'production';
|
const debug = process.env.NODE_ENV !== 'production';
|
||||||
|
|
||||||
|
const vuexLocal = new VuexPersistence();
|
||||||
|
|
||||||
export default createStore(Object.assign({}, root, {
|
export default createStore(Object.assign({}, root, {
|
||||||
modules: {
|
modules: {
|
||||||
uistate,
|
uistate,
|
||||||
@@ -15,5 +18,5 @@ export default createStore(Object.assign({}, root, {
|
|||||||
reader,
|
reader,
|
||||||
},
|
},
|
||||||
strict: debug,
|
strict: debug,
|
||||||
plugins: [createPersistedState()]
|
plugins: [vuexLocal.plugin]
|
||||||
}));
|
}));
|
||||||
|
|||||||
@@ -180,17 +180,28 @@ const settingDefaults = {
|
|||||||
|
|
||||||
showServerStorageMessages: true,
|
showServerStorageMessages: true,
|
||||||
showWhatsNewDialog: true,
|
showWhatsNewDialog: true,
|
||||||
showDonationDialog2020: true,
|
showDonationDialog: true,
|
||||||
showNeedUpdateNotify: true,
|
showNeedUpdateNotify: true,
|
||||||
|
|
||||||
fontShifts: {},
|
fontShifts: {},
|
||||||
showToolButton: {},
|
showToolButton: {},
|
||||||
toolBarHideOnScroll: true,
|
toolBarHideOnScroll: false,
|
||||||
userHotKeys: {},
|
userHotKeys: {},
|
||||||
userWallpapers: [],
|
userWallpapers: [],
|
||||||
|
|
||||||
recentShowSameBook: false,
|
recentShowSameBook: false,
|
||||||
recentSortMethod: '',
|
recentSortMethod: '',
|
||||||
|
|
||||||
|
//Book Update Checker
|
||||||
|
bucEnabled: true, // общее включение/выключение проверки обновлений
|
||||||
|
bucSizeDiff: 1, // разница в размерах файла, при которой показывать наличие обновления
|
||||||
|
bucSetOnNew: true, // автоматически включать проверку обновлений для вновь загружаемых файлов
|
||||||
|
bucCancelEnabled: true, // вкл/выкл отмену проверки книг через bucCancelDays
|
||||||
|
bucCancelDays: 90, // количество дней, через которое отменяется проверка книги, при условии отсутствия обновлений за это время
|
||||||
|
|
||||||
|
//для SettingsPage
|
||||||
|
needUpdateSettingsView: 0,
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
for (const font of fonts)
|
for (const font of fonts)
|
||||||
|
|||||||
@@ -6,6 +6,7 @@ server {
|
|||||||
ssl_dhparam /etc/letsencrypt/ssl-dhparams.pem; # managed by Certbot
|
ssl_dhparam /etc/letsencrypt/ssl-dhparams.pem; # managed by Certbot
|
||||||
|
|
||||||
server_name beta.liberama.top;
|
server_name beta.liberama.top;
|
||||||
|
set $liberama http://127.0.0.1:34082;
|
||||||
|
|
||||||
client_max_body_size 50m;
|
client_max_body_size 50m;
|
||||||
proxy_read_timeout 1h;
|
proxy_read_timeout 1h;
|
||||||
@@ -15,15 +16,20 @@ server {
|
|||||||
gzip_proxied expired no-cache no-store private auth;
|
gzip_proxied expired no-cache no-store private auth;
|
||||||
gzip_types *;
|
gzip_types *;
|
||||||
|
|
||||||
|
location @liberama {
|
||||||
|
proxy_pass $liberama;
|
||||||
|
}
|
||||||
|
|
||||||
location /api {
|
location /api {
|
||||||
proxy_pass http://127.0.0.1:34082;
|
proxy_pass $liberama;
|
||||||
}
|
}
|
||||||
|
|
||||||
location /ws {
|
location /ws {
|
||||||
proxy_pass http://127.0.0.1:34082;
|
proxy_pass $liberama;
|
||||||
proxy_http_version 1.1;
|
proxy_http_version 1.1;
|
||||||
proxy_set_header Upgrade $http_upgrade;
|
proxy_set_header Upgrade $http_upgrade;
|
||||||
proxy_set_header Connection "upgrade";
|
proxy_set_header Connection "upgrade";
|
||||||
|
proxy_read_timeout 600s;
|
||||||
}
|
}
|
||||||
|
|
||||||
location / {
|
location / {
|
||||||
@@ -32,6 +38,11 @@ server {
|
|||||||
location /tmp {
|
location /tmp {
|
||||||
types { } default_type "application/xml; charset=utf-8";
|
types { } default_type "application/xml; charset=utf-8";
|
||||||
add_header Content-Encoding gzip;
|
add_header Content-Encoding gzip;
|
||||||
|
try_files $uri @liberama;
|
||||||
|
}
|
||||||
|
|
||||||
|
location /upload {
|
||||||
|
try_files $uri @liberama;
|
||||||
}
|
}
|
||||||
|
|
||||||
location ~* \.(?:manifest|appcache|html)$ {
|
location ~* \.(?:manifest|appcache|html)$ {
|
||||||
@@ -50,6 +61,7 @@ server {
|
|||||||
server {
|
server {
|
||||||
listen 80;
|
listen 80;
|
||||||
server_name b.beta.liberama.top;
|
server_name b.beta.liberama.top;
|
||||||
|
set $liberama http://127.0.0.1:34082;
|
||||||
|
|
||||||
client_max_body_size 50m;
|
client_max_body_size 50m;
|
||||||
proxy_read_timeout 1h;
|
proxy_read_timeout 1h;
|
||||||
@@ -59,15 +71,20 @@ server {
|
|||||||
gzip_proxied expired no-cache no-store private auth;
|
gzip_proxied expired no-cache no-store private auth;
|
||||||
gzip_types *;
|
gzip_types *;
|
||||||
|
|
||||||
|
location @liberama {
|
||||||
|
proxy_pass $liberama;
|
||||||
|
}
|
||||||
|
|
||||||
location /api {
|
location /api {
|
||||||
proxy_pass http://127.0.0.1:34082;
|
proxy_pass $liberama;
|
||||||
}
|
}
|
||||||
|
|
||||||
location /ws {
|
location /ws {
|
||||||
proxy_pass http://127.0.0.1:34082;
|
proxy_pass $liberama;
|
||||||
proxy_http_version 1.1;
|
proxy_http_version 1.1;
|
||||||
proxy_set_header Upgrade $http_upgrade;
|
proxy_set_header Upgrade $http_upgrade;
|
||||||
proxy_set_header Connection "upgrade";
|
proxy_set_header Connection "upgrade";
|
||||||
|
proxy_read_timeout 600s;
|
||||||
}
|
}
|
||||||
|
|
||||||
location / {
|
location / {
|
||||||
@@ -76,6 +93,11 @@ server {
|
|||||||
location /tmp {
|
location /tmp {
|
||||||
types { } default_type "application/xml; charset=utf-8";
|
types { } default_type "application/xml; charset=utf-8";
|
||||||
add_header Content-Encoding gzip;
|
add_header Content-Encoding gzip;
|
||||||
|
try_files $uri @liberama;
|
||||||
|
}
|
||||||
|
|
||||||
|
location /upload {
|
||||||
|
try_files $uri @liberama;
|
||||||
}
|
}
|
||||||
|
|
||||||
location ~* \.(?:manifest|appcache|html)$ {
|
location ~* \.(?:manifest|appcache|html)$ {
|
||||||
|
|||||||
@@ -6,6 +6,7 @@ server {
|
|||||||
ssl_dhparam /etc/letsencrypt/ssl-dhparams.pem; # managed by Certbot
|
ssl_dhparam /etc/letsencrypt/ssl-dhparams.pem; # managed by Certbot
|
||||||
|
|
||||||
server_name beta.omnireader.ru;
|
server_name beta.omnireader.ru;
|
||||||
|
set $liberama http://127.0.0.1:34081;
|
||||||
|
|
||||||
client_max_body_size 50m;
|
client_max_body_size 50m;
|
||||||
proxy_read_timeout 1h;
|
proxy_read_timeout 1h;
|
||||||
@@ -15,15 +16,20 @@ server {
|
|||||||
gzip_proxied expired no-cache no-store private auth;
|
gzip_proxied expired no-cache no-store private auth;
|
||||||
gzip_types *;
|
gzip_types *;
|
||||||
|
|
||||||
|
location @liberama {
|
||||||
|
proxy_pass $liberama;
|
||||||
|
}
|
||||||
|
|
||||||
location /api {
|
location /api {
|
||||||
proxy_pass http://127.0.0.1:34081;
|
proxy_pass $liberama;
|
||||||
}
|
}
|
||||||
|
|
||||||
location /ws {
|
location /ws {
|
||||||
proxy_pass http://127.0.0.1:34081;
|
proxy_pass $liberama;
|
||||||
proxy_http_version 1.1;
|
proxy_http_version 1.1;
|
||||||
proxy_set_header Upgrade $http_upgrade;
|
proxy_set_header Upgrade $http_upgrade;
|
||||||
proxy_set_header Connection "upgrade";
|
proxy_set_header Connection "upgrade";
|
||||||
|
proxy_read_timeout 600s;
|
||||||
}
|
}
|
||||||
|
|
||||||
location / {
|
location / {
|
||||||
@@ -32,6 +38,11 @@ server {
|
|||||||
location /tmp {
|
location /tmp {
|
||||||
types { } default_type "application/xml; charset=utf-8";
|
types { } default_type "application/xml; charset=utf-8";
|
||||||
add_header Content-Encoding gzip;
|
add_header Content-Encoding gzip;
|
||||||
|
try_files $uri @liberama;
|
||||||
|
}
|
||||||
|
|
||||||
|
location /upload {
|
||||||
|
try_files $uri @liberama;
|
||||||
}
|
}
|
||||||
|
|
||||||
location ~* \.(?:manifest|appcache|html)$ {
|
location ~* \.(?:manifest|appcache|html)$ {
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
server {
|
server {
|
||||||
listen 80;
|
listen 80;
|
||||||
server_name beta.omnireader.ru;
|
server_name beta.omnireader.ru;
|
||||||
|
set $liberama http://127.0.0.1:34081;
|
||||||
|
|
||||||
client_max_body_size 50m;
|
client_max_body_size 50m;
|
||||||
proxy_read_timeout 1h;
|
proxy_read_timeout 1h;
|
||||||
@@ -10,15 +11,20 @@ server {
|
|||||||
gzip_proxied expired no-cache no-store private auth;
|
gzip_proxied expired no-cache no-store private auth;
|
||||||
gzip_types *;
|
gzip_types *;
|
||||||
|
|
||||||
|
location @liberama {
|
||||||
|
proxy_pass $liberama;
|
||||||
|
}
|
||||||
|
|
||||||
location /api {
|
location /api {
|
||||||
proxy_pass http://127.0.0.1:34081;
|
proxy_pass $liberama;
|
||||||
}
|
}
|
||||||
|
|
||||||
location /ws {
|
location /ws {
|
||||||
proxy_pass http://127.0.0.1:34081;
|
proxy_pass $liberama;
|
||||||
proxy_http_version 1.1;
|
proxy_http_version 1.1;
|
||||||
proxy_set_header Upgrade $http_upgrade;
|
proxy_set_header Upgrade $http_upgrade;
|
||||||
proxy_set_header Connection "upgrade";
|
proxy_set_header Connection "upgrade";
|
||||||
|
proxy_read_timeout 600s;
|
||||||
}
|
}
|
||||||
|
|
||||||
location / {
|
location / {
|
||||||
@@ -27,6 +33,11 @@ server {
|
|||||||
location /tmp {
|
location /tmp {
|
||||||
types { } default_type "application/xml; charset=utf-8";
|
types { } default_type "application/xml; charset=utf-8";
|
||||||
add_header Content-Encoding gzip;
|
add_header Content-Encoding gzip;
|
||||||
|
try_files $uri @liberama;
|
||||||
|
}
|
||||||
|
|
||||||
|
location /upload {
|
||||||
|
try_files $uri @liberama;
|
||||||
}
|
}
|
||||||
|
|
||||||
location ~* \.(?:manifest|appcache|html)$ {
|
location ~* \.(?:manifest|appcache|html)$ {
|
||||||
|
|||||||
@@ -17,6 +17,7 @@ server {
|
|||||||
ssl_dhparam /etc/letsencrypt/ssl-dhparams.pem; # managed by Certbot
|
ssl_dhparam /etc/letsencrypt/ssl-dhparams.pem; # managed by Certbot
|
||||||
|
|
||||||
server_name liberama.top;
|
server_name liberama.top;
|
||||||
|
set $liberama http://127.0.0.1:55081;
|
||||||
|
|
||||||
client_max_body_size 100m;
|
client_max_body_size 100m;
|
||||||
proxy_read_timeout 1h;
|
proxy_read_timeout 1h;
|
||||||
@@ -26,12 +27,16 @@ server {
|
|||||||
gzip_proxied expired no-cache no-store private auth;
|
gzip_proxied expired no-cache no-store private auth;
|
||||||
gzip_types *;
|
gzip_types *;
|
||||||
|
|
||||||
|
location @liberama {
|
||||||
|
proxy_pass $liberama;
|
||||||
|
}
|
||||||
|
|
||||||
location /api {
|
location /api {
|
||||||
proxy_pass http://127.0.0.1:55081;
|
proxy_pass $liberama;
|
||||||
}
|
}
|
||||||
|
|
||||||
location /ws {
|
location /ws {
|
||||||
proxy_pass http://127.0.0.1:55081;
|
proxy_pass $liberama;
|
||||||
proxy_http_version 1.1;
|
proxy_http_version 1.1;
|
||||||
proxy_set_header Upgrade $http_upgrade;
|
proxy_set_header Upgrade $http_upgrade;
|
||||||
proxy_set_header Connection "upgrade";
|
proxy_set_header Connection "upgrade";
|
||||||
@@ -44,6 +49,11 @@ server {
|
|||||||
location /tmp {
|
location /tmp {
|
||||||
types { } default_type "application/xml; charset=utf-8";
|
types { } default_type "application/xml; charset=utf-8";
|
||||||
add_header Content-Encoding gzip;
|
add_header Content-Encoding gzip;
|
||||||
|
try_files $uri @liberama;
|
||||||
|
}
|
||||||
|
|
||||||
|
location /upload {
|
||||||
|
try_files $uri @liberama;
|
||||||
}
|
}
|
||||||
|
|
||||||
location ~* \.(?:manifest|appcache|html)$ {
|
location ~* \.(?:manifest|appcache|html)$ {
|
||||||
@@ -62,6 +72,7 @@ server {
|
|||||||
server {
|
server {
|
||||||
listen 80;
|
listen 80;
|
||||||
server_name b.liberama.top;
|
server_name b.liberama.top;
|
||||||
|
set $liberama http://127.0.0.1:55081;
|
||||||
|
|
||||||
client_max_body_size 100m;
|
client_max_body_size 100m;
|
||||||
proxy_read_timeout 1h;
|
proxy_read_timeout 1h;
|
||||||
@@ -71,15 +82,20 @@ server {
|
|||||||
gzip_proxied expired no-cache no-store private auth;
|
gzip_proxied expired no-cache no-store private auth;
|
||||||
gzip_types *;
|
gzip_types *;
|
||||||
|
|
||||||
|
location @liberama {
|
||||||
|
proxy_pass $liberama;
|
||||||
|
}
|
||||||
|
|
||||||
location /api {
|
location /api {
|
||||||
proxy_pass http://127.0.0.1:55081;
|
proxy_pass $liberama;
|
||||||
}
|
}
|
||||||
|
|
||||||
location /ws {
|
location /ws {
|
||||||
proxy_pass http://127.0.0.1:55081;
|
proxy_pass $liberama;
|
||||||
proxy_http_version 1.1;
|
proxy_http_version 1.1;
|
||||||
proxy_set_header Upgrade $http_upgrade;
|
proxy_set_header Upgrade $http_upgrade;
|
||||||
proxy_set_header Connection "upgrade";
|
proxy_set_header Connection "upgrade";
|
||||||
|
proxy_read_timeout 600s;
|
||||||
}
|
}
|
||||||
|
|
||||||
location / {
|
location / {
|
||||||
@@ -88,6 +104,11 @@ server {
|
|||||||
location /tmp {
|
location /tmp {
|
||||||
types { } default_type "application/xml; charset=utf-8";
|
types { } default_type "application/xml; charset=utf-8";
|
||||||
add_header Content-Encoding gzip;
|
add_header Content-Encoding gzip;
|
||||||
|
try_files $uri @liberama;
|
||||||
|
}
|
||||||
|
|
||||||
|
location /upload {
|
||||||
|
try_files $uri @liberama;
|
||||||
}
|
}
|
||||||
|
|
||||||
location ~* \.(?:manifest|appcache|html)$ {
|
location ~* \.(?:manifest|appcache|html)$ {
|
||||||
|
|||||||
@@ -6,6 +6,7 @@ server {
|
|||||||
ssl_dhparam /etc/letsencrypt/ssl-dhparams.pem; # managed by Certbot
|
ssl_dhparam /etc/letsencrypt/ssl-dhparams.pem; # managed by Certbot
|
||||||
|
|
||||||
server_name omnireader.ru;
|
server_name omnireader.ru;
|
||||||
|
set $liberama http://127.0.0.1:44081;
|
||||||
|
|
||||||
client_max_body_size 100m;
|
client_max_body_size 100m;
|
||||||
proxy_read_timeout 1h;
|
proxy_read_timeout 1h;
|
||||||
@@ -15,12 +16,16 @@ server {
|
|||||||
gzip_proxied expired no-cache no-store private auth;
|
gzip_proxied expired no-cache no-store private auth;
|
||||||
gzip_types *;
|
gzip_types *;
|
||||||
|
|
||||||
|
location @liberama {
|
||||||
|
proxy_pass $liberama;
|
||||||
|
}
|
||||||
|
|
||||||
location /api {
|
location /api {
|
||||||
proxy_pass http://127.0.0.1:44081;
|
proxy_pass $liberama;
|
||||||
}
|
}
|
||||||
|
|
||||||
location /ws {
|
location /ws {
|
||||||
proxy_pass http://127.0.0.1:44081;
|
proxy_pass $liberama;
|
||||||
proxy_http_version 1.1;
|
proxy_http_version 1.1;
|
||||||
proxy_set_header Upgrade $http_upgrade;
|
proxy_set_header Upgrade $http_upgrade;
|
||||||
proxy_set_header Connection "upgrade";
|
proxy_set_header Connection "upgrade";
|
||||||
@@ -33,6 +38,11 @@ server {
|
|||||||
location /tmp {
|
location /tmp {
|
||||||
types { } default_type "application/xml; charset=utf-8";
|
types { } default_type "application/xml; charset=utf-8";
|
||||||
add_header Content-Encoding gzip;
|
add_header Content-Encoding gzip;
|
||||||
|
try_files $uri @liberama;
|
||||||
|
}
|
||||||
|
|
||||||
|
location /upload {
|
||||||
|
try_files $uri @liberama;
|
||||||
}
|
}
|
||||||
|
|
||||||
location ~* \.(?:manifest|appcache|html)$ {
|
location ~* \.(?:manifest|appcache|html)$ {
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
server {
|
server {
|
||||||
listen 80;
|
listen 80;
|
||||||
server_name omnireader.ru;
|
server_name omnireader.ru;
|
||||||
|
set $liberama http://127.0.0.1:44081;
|
||||||
|
|
||||||
client_max_body_size 50m;
|
client_max_body_size 50m;
|
||||||
proxy_read_timeout 1h;
|
proxy_read_timeout 1h;
|
||||||
@@ -10,12 +11,16 @@ server {
|
|||||||
gzip_proxied expired no-cache no-store private auth;
|
gzip_proxied expired no-cache no-store private auth;
|
||||||
gzip_types *;
|
gzip_types *;
|
||||||
|
|
||||||
|
location @liberama {
|
||||||
|
proxy_pass $liberama;
|
||||||
|
}
|
||||||
|
|
||||||
location /api {
|
location /api {
|
||||||
proxy_pass http://127.0.0.1:44081;
|
proxy_pass $liberama;
|
||||||
}
|
}
|
||||||
|
|
||||||
location /ws {
|
location /ws {
|
||||||
proxy_pass http://127.0.0.1:44081;
|
proxy_pass $liberama;
|
||||||
proxy_http_version 1.1;
|
proxy_http_version 1.1;
|
||||||
proxy_set_header Upgrade $http_upgrade;
|
proxy_set_header Upgrade $http_upgrade;
|
||||||
proxy_set_header Connection "upgrade";
|
proxy_set_header Connection "upgrade";
|
||||||
@@ -27,6 +32,11 @@ server {
|
|||||||
location /tmp {
|
location /tmp {
|
||||||
types { } default_type "application/xml; charset=utf-8";
|
types { } default_type "application/xml; charset=utf-8";
|
||||||
add_header Content-Encoding gzip;
|
add_header Content-Encoding gzip;
|
||||||
|
try_files $uri @liberama;
|
||||||
|
}
|
||||||
|
|
||||||
|
location /upload {
|
||||||
|
try_files $uri @liberama;
|
||||||
}
|
}
|
||||||
|
|
||||||
location ~* \.(?:manifest|appcache|html)$ {
|
location ~* \.(?:manifest|appcache|html)$ {
|
||||||
|
|||||||
5371
package-lock.json
generated
5371
package-lock.json
generated
File diff suppressed because it is too large
Load Diff
67
package.json
67
package.json
@@ -1,17 +1,17 @@
|
|||||||
{
|
{
|
||||||
"name": "Liberama",
|
"name": "Liberama",
|
||||||
"version": "0.11.7",
|
"version": "0.12.1",
|
||||||
"author": "Book Pauk <bookpauk@gmail.com>",
|
"author": "Book Pauk <bookpauk@gmail.com>",
|
||||||
"license": "CC0-1.0",
|
"license": "CC0-1.0",
|
||||||
"repository": "bookpauk/liberama",
|
"repository": "bookpauk/liberama",
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">=14.4.0"
|
"node": ">=16.16.0"
|
||||||
},
|
},
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"dev": "nodemon --inspect --ignore server/public --ignore server/data --ignore client --exec 'node server'",
|
"dev": "nodemon --inspect --ignore server/public --ignore server/data --ignore client --exec 'node server'",
|
||||||
"build:client": "webpack --config build/webpack.prod.config.js",
|
"build:client": "webpack --config build/webpack.prod.config.js",
|
||||||
"build:linux": "npm run build:client && node build/linux && pkg -t node14-linux-x64 -C GZip -o dist/linux/liberama .",
|
"build:linux": "npm run build:client && node build/linux && pkg -t node16-linux-x64 -C GZip -o dist/linux/liberama .",
|
||||||
"build:win": "npm run build:client && node build/win && pkg -t node14-win-x64 -C GZip -o dist/win/liberama .",
|
"build:win": "npm run build:client && node build/win && pkg -t node16-win-x64 -C GZip -o dist/win/liberama .",
|
||||||
"lint": "eslint --ext=.js,.vue client server",
|
"lint": "eslint --ext=.js,.vue client server",
|
||||||
"build:client-dev": "webpack --config build/webpack.dev.config.js",
|
"build:client-dev": "webpack --config build/webpack.dev.config.js",
|
||||||
"postinstall": "npm run build:client-dev && node build/linux"
|
"postinstall": "npm run build:client-dev && node build/linux"
|
||||||
@@ -21,67 +21,64 @@
|
|||||||
"scripts": "server/config/*.js"
|
"scripts": "server/config/*.js"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@babel/core": "^7.16.0",
|
"@babel/core": "^7.18.13",
|
||||||
"@babel/eslint-parser": "^7.16.3",
|
"@babel/eslint-parser": "^7.18.9",
|
||||||
"@babel/eslint-plugin": "^7.14.5",
|
"@babel/eslint-plugin": "^7.18.10",
|
||||||
"@babel/plugin-proposal-decorators": "^7.16.0",
|
"@babel/plugin-proposal-decorators": "^7.18.10",
|
||||||
"@babel/preset-env": "^7.16.0",
|
"@babel/preset-env": "^7.18.10",
|
||||||
"@vue/compiler-sfc": "^3.2.22",
|
"@vue/compiler-sfc": "^3.2.22",
|
||||||
"babel-loader": "^8.2.3",
|
"babel-loader": "^8.2.5",
|
||||||
"copy-webpack-plugin": "^11.0.0",
|
"copy-webpack-plugin": "^11.0.0",
|
||||||
"css-loader": "^6.5.1",
|
"css-loader": "^6.7.1",
|
||||||
"css-minimizer-webpack-plugin": "^4.0.0",
|
"css-minimizer-webpack-plugin": "^4.0.0",
|
||||||
"eslint": "^8.19.0",
|
"eslint": "^8.23.0",
|
||||||
"eslint-plugin-vue": "^9.2.0",
|
"eslint-plugin-vue": "^9.4.0",
|
||||||
"html-webpack-plugin": "^5.5.0",
|
"html-webpack-plugin": "^5.5.0",
|
||||||
"mini-css-extract-plugin": "^2.4.4",
|
"mini-css-extract-plugin": "^2.6.1",
|
||||||
"pkg": "^5.5.1",
|
"pkg": "^5.8.0",
|
||||||
"terser-webpack-plugin": "^5.2.5",
|
"terser-webpack-plugin": "^5.3.6",
|
||||||
"vue-eslint-parser": "^9.0.3",
|
"vue-eslint-parser": "^9.0.3",
|
||||||
"vue-loader": "^17.0.0",
|
"vue-loader": "^17.0.0",
|
||||||
"vue-style-loader": "^4.1.3",
|
"vue-style-loader": "^4.1.3",
|
||||||
"webpack": "^5.64.1",
|
"webpack": "^5.74.0",
|
||||||
"webpack-cli": "^4.9.1",
|
"webpack-cli": "^4.10.0",
|
||||||
"webpack-dev-middleware": "^5.2.1",
|
"webpack-dev-middleware": "^5.3.3",
|
||||||
"webpack-hot-middleware": "^2.25.1",
|
"webpack-hot-middleware": "^2.25.2",
|
||||||
"webpack-merge": "^5.8.0",
|
"webpack-merge": "^5.8.0",
|
||||||
"workbox-webpack-plugin": "^6.4.1"
|
"workbox-webpack-plugin": "^6.5.4"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@quasar/extras": "^1.12.0",
|
"@quasar/extras": "^1.15.2",
|
||||||
"@vue/compat": "^3.2.21",
|
"@vue/compat": "^3.2.38",
|
||||||
"axios": "^0.27.2",
|
"axios": "^0.27.2",
|
||||||
"base-x": "^4.0.0",
|
"base-x": "^4.0.0",
|
||||||
"chardet": "^1.4.0",
|
"chardet": "^1.4.0",
|
||||||
"compression": "^1.7.4",
|
"compression": "^1.7.4",
|
||||||
"express": "^4.17.1",
|
"express": "^4.18.1",
|
||||||
"fg-loadcss": "^3.1.0",
|
"fg-loadcss": "^3.1.0",
|
||||||
"fs-extra": "^10.1.0",
|
"fs-extra": "^10.1.0",
|
||||||
"he": "^1.2.0",
|
"he": "^1.2.0",
|
||||||
"iconv-lite": "^0.6.3",
|
"iconv-lite": "^0.6.3",
|
||||||
"jembadb": "^3.0.8",
|
"jembadb": "^4.2.0",
|
||||||
"localforage": "^1.10.0",
|
"localforage": "^1.10.0",
|
||||||
"lodash": "^4.17.21",
|
"lodash": "^4.17.21",
|
||||||
"minimist": "^1.2.5",
|
"minimist": "^1.2.6",
|
||||||
"multer": "^1.4.5-lts.1",
|
"multer": "^1.4.5-lts.1",
|
||||||
"pako": "^2.0.4",
|
"pako": "^2.0.4",
|
||||||
"path-browserify": "^1.0.1",
|
"path-browserify": "^1.0.1",
|
||||||
"pidusage": "^3.0.0",
|
"pidusage": "^3.0.0",
|
||||||
"quasar": "^2.7.5",
|
"quasar": "^2.7.7",
|
||||||
"safe-buffer": "^5.2.1",
|
"safe-buffer": "^5.2.1",
|
||||||
"sanitize-html": "^2.5.3",
|
"sanitize-html": "^2.7.1",
|
||||||
"sjcl": "^1.0.8",
|
"sjcl": "^1.0.8",
|
||||||
"sql-template-strings": "^2.2.2",
|
|
||||||
"sqlite": "^4.0.23",
|
|
||||||
"sqlite3": "^5.0.2",
|
|
||||||
"tar-fs": "^2.1.1",
|
"tar-fs": "^2.1.1",
|
||||||
"unbzip2-stream": "^1.4.3",
|
"unbzip2-stream": "^1.4.3",
|
||||||
"vue": "^3.2.37",
|
"vue": "^3.2.37",
|
||||||
"vue-router": "^4.1.1",
|
"vue-router": "^4.1.5",
|
||||||
"vuex": "^4.0.2",
|
"vuex": "^4.0.2",
|
||||||
"vuex-persistedstate": "^4.1.0",
|
"vuex-persist": "^3.1.3",
|
||||||
"webdav": "^4.7.0",
|
"webdav": "^4.11.0",
|
||||||
"ws": "^8.2.3",
|
"ws": "^8.8.1",
|
||||||
"zip-stream": "^4.1.0"
|
"zip-stream": "^4.1.0"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -23,46 +23,61 @@ module.exports = {
|
|||||||
|
|
||||||
useExternalBookConverter: false,
|
useExternalBookConverter: false,
|
||||||
acceptFileExt: '.fb2, .fb3, .html, .txt, .zip, .bz2, .gz, .rar, .epub, .mobi, .rtf, .doc, .docx, .pdf, .djvu, .jpg, .jpeg, .png',
|
acceptFileExt: '.fb2, .fb3, .html, .txt, .zip, .bz2, .gz, .rar, .epub, .mobi, .rtf, .doc, .docx, .pdf, .djvu, .jpg, .jpeg, .png',
|
||||||
webConfigParams: ['name', 'version', 'mode', 'maxUploadFileSize', 'useExternalBookConverter', 'acceptFileExt', 'branch'],
|
webConfigParams: ['name', 'version', 'mode', 'maxUploadFileSize', 'useExternalBookConverter', 'acceptFileExt', 'bucEnabled', 'branch'],
|
||||||
|
|
||||||
db: [
|
|
||||||
{
|
|
||||||
poolName: 'app',
|
|
||||||
connCount: 20,
|
|
||||||
fileName: 'app.sqlite',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
poolName: 'readerStorage',
|
|
||||||
connCount: 20,
|
|
||||||
fileName: 'reader-storage.sqlite',
|
|
||||||
}
|
|
||||||
],
|
|
||||||
|
|
||||||
jembaDb: [
|
jembaDb: [
|
||||||
{
|
{
|
||||||
|
serverMode: ['reader', 'omnireader', 'liberama.top'],
|
||||||
|
dbName: 'app',
|
||||||
|
thread: true,
|
||||||
|
openAll: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
serverMode: ['reader', 'omnireader', 'liberama.top'],
|
||||||
dbName: 'reader-storage',
|
dbName: 'reader-storage',
|
||||||
thread: true,
|
thread: true,
|
||||||
openAll: true,
|
openAll: true,
|
||||||
}
|
},
|
||||||
|
{
|
||||||
|
serverMode: 'book_update_checker',
|
||||||
|
dbName: 'book-update-server',
|
||||||
|
thread: true,
|
||||||
|
openAll: true,
|
||||||
|
},
|
||||||
],
|
],
|
||||||
|
|
||||||
servers: [
|
servers: [
|
||||||
{
|
{
|
||||||
serverName: '1',
|
serverName: '1',
|
||||||
mode: 'normal', //'none', 'normal', 'site', 'reader', 'omnireader', 'liberama.top'
|
mode: 'normal', //'none', 'normal', 'site', 'reader', 'omnireader', 'liberama.top', 'book_update_checker'
|
||||||
ip: '0.0.0.0',
|
ip: '0.0.0.0',
|
||||||
port: '33080',
|
port: '33080',
|
||||||
},
|
},
|
||||||
|
/*{
|
||||||
|
serverName: '2',
|
||||||
|
mode: 'book_update_checker', //'none', 'normal', 'site', 'reader', 'omnireader', 'liberama.top', 'book_update_checker'
|
||||||
|
isHttps: true,
|
||||||
|
keysFile: 'server',
|
||||||
|
ip: '0.0.0.0',
|
||||||
|
port: '33443',
|
||||||
|
accessToken: '',
|
||||||
|
}*/
|
||||||
],
|
],
|
||||||
|
|
||||||
remoteWebDavStorage: false,
|
remoteStorage: false,
|
||||||
/*
|
/*
|
||||||
remoteWebDavStorage: {
|
remoteStorage: {
|
||||||
url: '127.0.0.1:1900',
|
url: 'wss://127.0.0.1:11900',
|
||||||
username: '',
|
accessToken: '',
|
||||||
password: '',
|
|
||||||
},
|
},
|
||||||
*/
|
*/
|
||||||
|
bucEnabled: false,
|
||||||
|
bucServer: false,
|
||||||
|
/*
|
||||||
|
bucServer: {
|
||||||
|
url: 'wss://127.0.0.1:33443',
|
||||||
|
accessToken: '',
|
||||||
|
}
|
||||||
|
*/
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -10,7 +10,9 @@ const propsToSave = [
|
|||||||
'useExternalBookConverter',
|
'useExternalBookConverter',
|
||||||
|
|
||||||
'servers',
|
'servers',
|
||||||
'remoteWebDavStorage',
|
'remoteStorage',
|
||||||
|
'bucEnabled',
|
||||||
|
'bucServer',
|
||||||
];
|
];
|
||||||
|
|
||||||
let instance = null;
|
let instance = null;
|
||||||
|
|||||||
126
server/controllers/BookUpdateCheckerController.js
Normal file
126
server/controllers/BookUpdateCheckerController.js
Normal file
@@ -0,0 +1,126 @@
|
|||||||
|
const WebSocket = require('ws');
|
||||||
|
//const _ = require('lodash');
|
||||||
|
|
||||||
|
const BUCServer = require('../core/BookUpdateChecker/BUCServer');
|
||||||
|
const log = new (require('../core/AppLogger'))().log;//singleton
|
||||||
|
//const utils = require('../core/utils');
|
||||||
|
|
||||||
|
const cleanPeriod = 1*60*1000;//1 минута
|
||||||
|
const closeSocketOnIdle = 5*60*1000;//5 минут
|
||||||
|
|
||||||
|
class BookUpdateCheckerController {
|
||||||
|
constructor(wss, config) {
|
||||||
|
this.config = config;
|
||||||
|
this.isDevelopment = (config.branch == 'development');
|
||||||
|
|
||||||
|
this.accessToken = config.accessToken;
|
||||||
|
this.bucServer = new BUCServer(config);
|
||||||
|
|
||||||
|
this.wss = wss;
|
||||||
|
|
||||||
|
wss.on('connection', (ws) => {
|
||||||
|
ws.on('message', (message) => {
|
||||||
|
this.onMessage(ws, message.toString());
|
||||||
|
});
|
||||||
|
|
||||||
|
ws.on('error', (err) => {
|
||||||
|
log(LM_ERR, err);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
setTimeout(() => { this.periodicClean(); }, cleanPeriod);
|
||||||
|
}
|
||||||
|
|
||||||
|
periodicClean() {
|
||||||
|
try {
|
||||||
|
const now = Date.now();
|
||||||
|
this.wss.clients.forEach((ws) => {
|
||||||
|
if (!ws.lastActivity || now - ws.lastActivity > closeSocketOnIdle - 50) {
|
||||||
|
ws.terminate();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
} finally {
|
||||||
|
setTimeout(() => { this.periodicClean(); }, cleanPeriod);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async onMessage(ws, message) {
|
||||||
|
let req = {};
|
||||||
|
try {
|
||||||
|
if (this.isDevelopment) {
|
||||||
|
log(`BUC-WebSocket-IN: ${message.substr(0, 4000)}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
req = JSON.parse(message);
|
||||||
|
|
||||||
|
ws.lastActivity = Date.now();
|
||||||
|
|
||||||
|
//pong for WebSocketConnection
|
||||||
|
this.send({_rok: 1}, req, ws);
|
||||||
|
|
||||||
|
if (req.accessToken !== this.accessToken)
|
||||||
|
throw new Error('Access denied');
|
||||||
|
|
||||||
|
switch (req.action) {
|
||||||
|
case 'test':
|
||||||
|
await this.test(req, ws); break;
|
||||||
|
case 'get-buc':
|
||||||
|
await this.getBuc(req, ws); break;
|
||||||
|
case 'update-buc':
|
||||||
|
await this.updateBuc(req, ws); break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
throw new Error(`Action not found: ${req.action}`);
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
this.send({error: e.message}, req, ws);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
send(res, req, ws) {
|
||||||
|
if (ws.readyState == WebSocket.OPEN) {
|
||||||
|
ws.lastActivity = Date.now();
|
||||||
|
let r = res;
|
||||||
|
if (req.requestId)
|
||||||
|
r = Object.assign({requestId: req.requestId}, r);
|
||||||
|
|
||||||
|
const message = JSON.stringify(r);
|
||||||
|
ws.send(message);
|
||||||
|
|
||||||
|
if (this.isDevelopment) {
|
||||||
|
log(`BUC-WebSocket-OUT: ${message.substr(0, 4000)}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//Actions ------------------------------------------------------------------
|
||||||
|
async test(req, ws) {
|
||||||
|
this.send({message: 'Liberama project is awesome'}, req, ws);
|
||||||
|
}
|
||||||
|
|
||||||
|
async getBuc(req, ws) {
|
||||||
|
if (!req.fromCheckTime)
|
||||||
|
throw new Error(`key 'fromCheckTime' is empty`);
|
||||||
|
|
||||||
|
await this.bucServer.getBuc(req.fromCheckTime, (rows) => {
|
||||||
|
this.send({state: 'get', rows}, req, ws);
|
||||||
|
});
|
||||||
|
|
||||||
|
this.send({state: 'finish'}, req, ws);
|
||||||
|
}
|
||||||
|
|
||||||
|
async updateBuc(req, ws) {
|
||||||
|
if (!req.bookUrls)
|
||||||
|
throw new Error(`key 'bookUrls' is empty`);
|
||||||
|
|
||||||
|
if (!Array.isArray(req.bookUrls))
|
||||||
|
throw new Error(`key 'bookUrls' must be array`);
|
||||||
|
|
||||||
|
await this.bucServer.updateBuc(req.bookUrls);
|
||||||
|
|
||||||
|
this.send({state: 'success'}, req, ws);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = BookUpdateCheckerController;
|
||||||
@@ -68,24 +68,6 @@ class ReaderController extends BaseController {
|
|||||||
res.status(400).send({error});
|
res.status(400).send({error});
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
async restoreCachedFile(req, res) {
|
|
||||||
const request = req.body;
|
|
||||||
let error = '';
|
|
||||||
try {
|
|
||||||
if (!request.path)
|
|
||||||
throw new Error(`key 'path' is empty`);
|
|
||||||
|
|
||||||
const workerId = this.readerWorker.restoreCachedFile(request.path);
|
|
||||||
const state = this.workerState.getState(workerId);
|
|
||||||
return (state ? state : {});
|
|
||||||
} catch (e) {
|
|
||||||
error = e.message;
|
|
||||||
}
|
|
||||||
//bad request
|
|
||||||
res.status(400).send({error});
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
module.exports = ReaderController;
|
module.exports = ReaderController;
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ const _ = require('lodash');
|
|||||||
const ReaderWorker = require('../core/Reader/ReaderWorker');//singleton
|
const ReaderWorker = require('../core/Reader/ReaderWorker');//singleton
|
||||||
const JembaReaderStorage = require('../core/Reader/JembaReaderStorage');//singleton
|
const JembaReaderStorage = require('../core/Reader/JembaReaderStorage');//singleton
|
||||||
const WorkerState = require('../core/WorkerState');//singleton
|
const WorkerState = require('../core/WorkerState');//singleton
|
||||||
|
const BUCClient = require('../core/BookUpdateChecker/BUCClient');//singleton
|
||||||
const log = new (require('../core/AppLogger'))().log;//singleton
|
const log = new (require('../core/AppLogger'))().log;//singleton
|
||||||
const utils = require('../core/utils');
|
const utils = require('../core/utils');
|
||||||
|
|
||||||
@@ -19,12 +20,20 @@ class WebSocketController {
|
|||||||
this.readerWorker = new ReaderWorker(config);
|
this.readerWorker = new ReaderWorker(config);
|
||||||
this.workerState = new WorkerState();
|
this.workerState = new WorkerState();
|
||||||
|
|
||||||
|
if (config.bucEnabled) {
|
||||||
|
this.bucClient = new BUCClient(config);
|
||||||
|
}
|
||||||
|
|
||||||
this.wss = wss;
|
this.wss = wss;
|
||||||
|
|
||||||
wss.on('connection', (ws) => {
|
wss.on('connection', (ws) => {
|
||||||
ws.on('message', (message) => {
|
ws.on('message', (message) => {
|
||||||
this.onMessage(ws, message.toString());
|
this.onMessage(ws, message.toString());
|
||||||
});
|
});
|
||||||
|
|
||||||
|
ws.on('error', (err) => {
|
||||||
|
log(LM_ERR, err);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
setTimeout(() => { this.periodicClean(); }, cleanPeriod);
|
setTimeout(() => { this.periodicClean(); }, cleanPeriod);
|
||||||
@@ -66,10 +75,14 @@ class WebSocketController {
|
|||||||
await this.workerGetState(req, ws); break;
|
await this.workerGetState(req, ws); break;
|
||||||
case 'worker-get-state-finish':
|
case 'worker-get-state-finish':
|
||||||
await this.workerGetStateFinish(req, ws); break;
|
await this.workerGetStateFinish(req, ws); break;
|
||||||
case 'reader-restore-cached-file':
|
|
||||||
await this.readerRestoreCachedFile(req, ws); break;
|
|
||||||
case 'reader-storage':
|
case 'reader-storage':
|
||||||
await this.readerStorageDo(req, ws); break;
|
await this.readerStorageDo(req, ws); break;
|
||||||
|
case 'upload-file-buf':
|
||||||
|
await this.uploadFileBuf(req, ws); break;
|
||||||
|
case 'upload-file-touch':
|
||||||
|
await this.uploadFileTouch(req, ws); break;
|
||||||
|
case 'check-buc':
|
||||||
|
await this.checkBuc(req, ws); break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
throw new Error(`Action not found: ${req.action}`);
|
throw new Error(`Action not found: ${req.action}`);
|
||||||
@@ -149,15 +162,6 @@ class WebSocketController {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async readerRestoreCachedFile(req, ws) {
|
|
||||||
if (!req.path)
|
|
||||||
throw new Error(`key 'path' is empty`);
|
|
||||||
|
|
||||||
const workerId = this.readerWorker.restoreCachedFile(req.path);
|
|
||||||
const state = this.workerState.getState(workerId);
|
|
||||||
this.send((state ? state : {}), req, ws);
|
|
||||||
}
|
|
||||||
|
|
||||||
async readerStorageDo(req, ws) {
|
async readerStorageDo(req, ws) {
|
||||||
if (!req.body)
|
if (!req.body)
|
||||||
throw new Error(`key 'body' is empty`);
|
throw new Error(`key 'body' is empty`);
|
||||||
@@ -168,6 +172,35 @@ class WebSocketController {
|
|||||||
|
|
||||||
this.send(await this.readerStorage.doAction(req.body), req, ws);
|
this.send(await this.readerStorage.doAction(req.body), req, ws);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async uploadFileBuf(req, ws) {
|
||||||
|
if (!req.buf)
|
||||||
|
throw new Error(`key 'buf' is empty`);
|
||||||
|
|
||||||
|
this.send({url: await this.readerWorker.saveFileBuf(req.buf)}, req, ws);
|
||||||
|
}
|
||||||
|
|
||||||
|
async uploadFileTouch(req, ws) {
|
||||||
|
if (!req.url)
|
||||||
|
throw new Error(`key 'url' is empty`);
|
||||||
|
|
||||||
|
this.send({url: await this.readerWorker.uploadFileTouch(req.url)}, req, ws);
|
||||||
|
}
|
||||||
|
|
||||||
|
async checkBuc(req, ws) {
|
||||||
|
if (!this.config.bucEnabled)
|
||||||
|
throw new Error('BookUpdateChecker disabled');
|
||||||
|
|
||||||
|
if (!req.bookUrls)
|
||||||
|
throw new Error(`key 'bookUrls' is empty`);
|
||||||
|
|
||||||
|
if (!Array.isArray(req.bookUrls))
|
||||||
|
throw new Error(`key 'bookUrls' must be array`);
|
||||||
|
|
||||||
|
const data = await this.bucClient.checkBuc(req.bookUrls);
|
||||||
|
|
||||||
|
this.send({state: 'success', data}, req, ws);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
module.exports = WebSocketController;
|
module.exports = WebSocketController;
|
||||||
|
|||||||
@@ -3,4 +3,5 @@ module.exports = {
|
|||||||
ReaderController: require('./ReaderController'),
|
ReaderController: require('./ReaderController'),
|
||||||
WorkerController: require('./WorkerController'),
|
WorkerController: require('./WorkerController'),
|
||||||
WebSocketController: require('./WebSocketController'),
|
WebSocketController: require('./WebSocketController'),
|
||||||
|
BookUpdateCheckerController: require('./BookUpdateCheckerController'),
|
||||||
}
|
}
|
||||||
262
server/core/BookUpdateChecker/BUCClient.js
Normal file
262
server/core/BookUpdateChecker/BUCClient.js
Normal file
@@ -0,0 +1,262 @@
|
|||||||
|
const WebSocketConnection = require('../WebSocketConnection');
|
||||||
|
const JembaConnManager = require('../../db/JembaConnManager');//singleton
|
||||||
|
|
||||||
|
const ayncExit = new (require('../AsyncExit'))();
|
||||||
|
const utils = require('../utils');
|
||||||
|
const log = new (require('../AppLogger'))().log;//singleton
|
||||||
|
|
||||||
|
const minuteMs = 60*1000;
|
||||||
|
const hourMs = 60*minuteMs;
|
||||||
|
const dayMs = 24*hourMs;
|
||||||
|
|
||||||
|
let instance = null;
|
||||||
|
|
||||||
|
//singleton
|
||||||
|
class BUCClient {
|
||||||
|
constructor(config) {
|
||||||
|
if (!instance) {
|
||||||
|
this.config = config;
|
||||||
|
|
||||||
|
this.connManager = new JembaConnManager();
|
||||||
|
this.appDb = this.connManager.db['app'];
|
||||||
|
|
||||||
|
this.wsc = new WebSocketConnection(config.bucServer.url, 10, 30, {rejectUnauthorized: false});
|
||||||
|
this.accessToken = config.bucServer.accessToken;
|
||||||
|
|
||||||
|
//константы
|
||||||
|
if (this.config.branch !== 'development') {
|
||||||
|
this.cleanQueryInterval = 300*dayMs;//интервал очистки устаревших
|
||||||
|
this.syncPeriod = 1*hourMs;//период синхронизации с сервером BUC
|
||||||
|
this.sendBookUrlsPeriod = 1*minuteMs;//период отправки BookUrls на сервер BUC
|
||||||
|
} else {
|
||||||
|
this.cleanQueryInterval = 300*dayMs;//интервал очистки устаревших
|
||||||
|
this.syncPeriod = 1*minuteMs;//период синхронизации с сервером BUC
|
||||||
|
this.sendBookUrlsPeriod = 1*1000;//период отправки BookUrls на сервер BUC
|
||||||
|
}
|
||||||
|
|
||||||
|
this.fromCheckTime = 1;
|
||||||
|
this.bookUrls = new Set();
|
||||||
|
|
||||||
|
this.main();//no await
|
||||||
|
|
||||||
|
instance = this;
|
||||||
|
}
|
||||||
|
|
||||||
|
return instance;
|
||||||
|
}
|
||||||
|
|
||||||
|
async wsRequest(query) {
|
||||||
|
const response = await this.wsc.message(
|
||||||
|
await this.wsc.send(Object.assign({accessToken: this.accessToken}, query), 60),
|
||||||
|
60
|
||||||
|
);
|
||||||
|
if (response.error)
|
||||||
|
throw new Error(response.error);
|
||||||
|
return response;
|
||||||
|
}
|
||||||
|
|
||||||
|
async wsGetBuc(fromCheckTime, callback) {
|
||||||
|
const requestId = await this.wsc.send({accessToken: this.accessToken, action: 'get-buc', fromCheckTime}, 60);
|
||||||
|
while (1) {//eslint-disable-line
|
||||||
|
const res = await this.wsc.message(requestId, 60);
|
||||||
|
|
||||||
|
if (res.state == 'get') {
|
||||||
|
await callback(res.rows);
|
||||||
|
} else {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async wsUpdateBuc(bookUrls) {
|
||||||
|
return await this.wsRequest({action: 'update-buc', bookUrls});
|
||||||
|
}
|
||||||
|
|
||||||
|
async checkBuc(bookUrls) {
|
||||||
|
const db = this.appDb;
|
||||||
|
|
||||||
|
for (const url of bookUrls)
|
||||||
|
this.bookUrls.add(url);
|
||||||
|
|
||||||
|
const rows = await db.select({
|
||||||
|
table: 'buc',
|
||||||
|
map: `(r) => ({id: r.id, size: r.size})`,
|
||||||
|
where: `@@id(${db.esc(bookUrls)})`,
|
||||||
|
});
|
||||||
|
|
||||||
|
return rows;
|
||||||
|
}
|
||||||
|
|
||||||
|
async findMaxCheckTime() {
|
||||||
|
const db = this.appDb;
|
||||||
|
|
||||||
|
let result = 1;
|
||||||
|
|
||||||
|
//одним куском, возможно будет жрать память
|
||||||
|
const rows = await db.select({
|
||||||
|
table: 'buc',
|
||||||
|
where: `
|
||||||
|
const result = new Set();
|
||||||
|
let max = 0;
|
||||||
|
let maxId = null;
|
||||||
|
|
||||||
|
@iter(@all(), (row) => {
|
||||||
|
if (row.checkTime > max) {
|
||||||
|
max = row.checkTime;
|
||||||
|
maxId = row.id;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
if (maxId)
|
||||||
|
result.add(maxId);
|
||||||
|
|
||||||
|
return result;
|
||||||
|
`
|
||||||
|
});
|
||||||
|
|
||||||
|
if (rows.length)
|
||||||
|
result = rows[0].checkTime;
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
async periodicSendBookUrls() {
|
||||||
|
while (1) {//eslint-disable-line
|
||||||
|
try {
|
||||||
|
//отправим this.bookUrls
|
||||||
|
if (this.bookUrls.size) {
|
||||||
|
log(`client: remote update buc begin`);
|
||||||
|
|
||||||
|
const arr = Array.from(this.bookUrls);
|
||||||
|
this.bookUrls = new Set();
|
||||||
|
|
||||||
|
const chunkSize = 100;
|
||||||
|
let updated = 0;
|
||||||
|
for (let i = 0; i < arr.length; i += chunkSize) {
|
||||||
|
const chunk = arr.slice(i, i + chunkSize);
|
||||||
|
|
||||||
|
const res = await this.wsUpdateBuc(chunk);
|
||||||
|
if (!res.error && res.state == 'success') {
|
||||||
|
//update success
|
||||||
|
updated += chunk.length;
|
||||||
|
} else {
|
||||||
|
for (const url of chunk) {
|
||||||
|
this.bookUrls.add(url);
|
||||||
|
}
|
||||||
|
log(LM_ERR, `update-buc error: ${(res.error ? res.error : `wrong state "${res.state}"`)}`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
log(`client: remote update buc end, updated ${updated} urls`);
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
log(LM_ERR, e.stack);
|
||||||
|
}
|
||||||
|
|
||||||
|
await utils.sleep(this.sendBookUrlsPeriod);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async periodicSync() {
|
||||||
|
const db = this.appDb;
|
||||||
|
|
||||||
|
while (1) {//eslint-disable-line
|
||||||
|
try {
|
||||||
|
//почистим нашу таблицу 'buc'
|
||||||
|
log(`client: clean 'buc' table begin`);
|
||||||
|
const cleanTime = Date.now() - this.cleanQueryInterval;
|
||||||
|
while (1) {//eslint-disable-line
|
||||||
|
//выборка всех по кусочкам
|
||||||
|
const rows = await db.select({
|
||||||
|
table: 'buc',
|
||||||
|
where: `
|
||||||
|
let iter = @getItem('clean');
|
||||||
|
if (!iter) {
|
||||||
|
iter = @all();
|
||||||
|
@setItem('clean', iter);
|
||||||
|
}
|
||||||
|
|
||||||
|
const ids = new Set();
|
||||||
|
let id = iter.next();
|
||||||
|
while (!id.done) {
|
||||||
|
ids.add(id.value);
|
||||||
|
if (ids.size >= 1000)
|
||||||
|
break;
|
||||||
|
id = iter.next();
|
||||||
|
}
|
||||||
|
|
||||||
|
return ids;
|
||||||
|
`
|
||||||
|
});
|
||||||
|
|
||||||
|
if (rows.length) {
|
||||||
|
const toDelIds = [];
|
||||||
|
for (const row of rows)
|
||||||
|
if (row.queryTime <= cleanTime)
|
||||||
|
toDelIds.push(row.id);
|
||||||
|
|
||||||
|
//удаление
|
||||||
|
const res = await db.delete({
|
||||||
|
table: 'buc',
|
||||||
|
where: `@@id(${db.esc(toDelIds)})`,
|
||||||
|
});
|
||||||
|
|
||||||
|
log(`client: clean 'buc' deleted ${res.deleted}`);
|
||||||
|
} else {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
await db.select({
|
||||||
|
table: 'buc',
|
||||||
|
where: `
|
||||||
|
@delItem('clean');
|
||||||
|
return new Set();
|
||||||
|
`
|
||||||
|
});
|
||||||
|
|
||||||
|
log(`client: clean 'buc' table end`);
|
||||||
|
|
||||||
|
//синхронизация с сервером BUC
|
||||||
|
log(`client: sync 'buc' table begin`);
|
||||||
|
this.fromCheckTime -= 30*minuteMs;//минус полчаса на всякий случай
|
||||||
|
await this.wsGetBuc(this.fromCheckTime, async(rows) => {
|
||||||
|
for (const row of rows) {
|
||||||
|
if (row.checkTime > this.fromCheckTime)
|
||||||
|
this.fromCheckTime = row.checkTime;
|
||||||
|
}
|
||||||
|
|
||||||
|
const res = await db.insert({
|
||||||
|
table: 'buc',
|
||||||
|
replace: true,
|
||||||
|
rows
|
||||||
|
});
|
||||||
|
|
||||||
|
log(`client: sync 'buc' table, inserted ${res.inserted} rows, replaced ${res.replaced}`);
|
||||||
|
});
|
||||||
|
log(`client: sync 'buc' table end`);
|
||||||
|
} catch (e) {
|
||||||
|
log(LM_ERR, e.stack);
|
||||||
|
}
|
||||||
|
|
||||||
|
await utils.sleep(this.syncPeriod);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async main() {
|
||||||
|
try {
|
||||||
|
if (!this.config.bucEnabled)
|
||||||
|
throw new Error('BookUpdateChecker disabled');
|
||||||
|
|
||||||
|
this.fromCheckTime = await this.findMaxCheckTime();
|
||||||
|
|
||||||
|
this.periodicSendBookUrls();//no await
|
||||||
|
this.periodicSync();//no await
|
||||||
|
|
||||||
|
log(`BUC Client Worker started`);
|
||||||
|
} catch (e) {
|
||||||
|
log(LM_FATAL, e.stack);
|
||||||
|
ayncExit.exit(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = BUCClient;
|
||||||
355
server/core/BookUpdateChecker/BUCServer.js
Normal file
355
server/core/BookUpdateChecker/BUCServer.js
Normal file
@@ -0,0 +1,355 @@
|
|||||||
|
const fs = require('fs-extra');
|
||||||
|
|
||||||
|
const FileDownloader = require('../FileDownloader');
|
||||||
|
const JembaConnManager = require('../../db/JembaConnManager');//singleton
|
||||||
|
|
||||||
|
const ayncExit = new (require('../AsyncExit'))();
|
||||||
|
const utils = require('../utils');
|
||||||
|
const log = new (require('../AppLogger'))().log;//singleton
|
||||||
|
|
||||||
|
const minuteMs = 60*1000;
|
||||||
|
const hourMs = 60*minuteMs;
|
||||||
|
const dayMs = 24*hourMs;
|
||||||
|
|
||||||
|
let instance = null;
|
||||||
|
|
||||||
|
//singleton
|
||||||
|
class BUCServer {
|
||||||
|
constructor(config) {
|
||||||
|
if (!instance) {
|
||||||
|
this.config = config;
|
||||||
|
|
||||||
|
//константы
|
||||||
|
if (this.config.branch !== 'development') {
|
||||||
|
this.maxCheckQueueLength = 10000;//максимальная длина checkQueue
|
||||||
|
this.fillCheckQueuePeriod = 1*minuteMs;//период пополнения очереди
|
||||||
|
this.periodicCheckWait = 500;//пауза, если нечего делать
|
||||||
|
|
||||||
|
this.cleanQueryInterval = 300*dayMs;//интервал очистки устаревших
|
||||||
|
this.oldQueryInterval = 14*dayMs;//интервал устаревания запроса на обновление
|
||||||
|
this.checkingInterval = 5*hourMs;//интервал проверки обновления одного и того же файла
|
||||||
|
this.sameHostCheckInterval = 1000;//интервал проверки файла на том же сайте, не менее
|
||||||
|
} else {
|
||||||
|
this.maxCheckQueueLength = 10;//максимальная длина checkQueue
|
||||||
|
this.fillCheckQueuePeriod = 10*1000;//период пополнения очереди
|
||||||
|
this.periodicCheckWait = 500;//пауза, если нечего делать
|
||||||
|
|
||||||
|
this.cleanQueryInterval = 300*dayMs;//интервал очистки устаревших
|
||||||
|
this.oldQueryInterval = 30*dayMs;//интервал устаревания запроса на обновление
|
||||||
|
this.checkingInterval = 30*1000;//интервал проверки обновления одного и того же файла
|
||||||
|
this.sameHostCheckInterval = 1000;//интервал проверки файла на том же сайте, не менее
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
this.config.tempDownloadDir = `${config.tempDir}/download`;
|
||||||
|
fs.ensureDirSync(this.config.tempDownloadDir);
|
||||||
|
|
||||||
|
this.down = new FileDownloader(config.maxUploadFileSize);
|
||||||
|
|
||||||
|
this.connManager = new JembaConnManager();
|
||||||
|
this.db = this.connManager.db['book-update-server'];
|
||||||
|
|
||||||
|
this.checkQueue = [];
|
||||||
|
this.hostChecking = {};
|
||||||
|
|
||||||
|
this.main(); //no await
|
||||||
|
|
||||||
|
instance = this;
|
||||||
|
}
|
||||||
|
|
||||||
|
return instance;
|
||||||
|
}
|
||||||
|
|
||||||
|
async getBuc(fromCheckTime, callback) {
|
||||||
|
const db = this.db;
|
||||||
|
|
||||||
|
const iterName = utils.randomHexString(30);
|
||||||
|
|
||||||
|
while (1) {//eslint-disable-line
|
||||||
|
const rows = await db.select({
|
||||||
|
table: 'buc',
|
||||||
|
where: `
|
||||||
|
let iter = @getItem(${db.esc(iterName)});
|
||||||
|
if (!iter) {
|
||||||
|
iter = @dirtyIndexLR('checkTime', ${db.esc(fromCheckTime)});
|
||||||
|
iter = iter.values();
|
||||||
|
@setItem(${db.esc(iterName)}, iter);
|
||||||
|
}
|
||||||
|
|
||||||
|
const ids = new Set();
|
||||||
|
let id = iter.next();
|
||||||
|
while (!id.done) {
|
||||||
|
ids.add(id.value);
|
||||||
|
if (ids.size >= 100)
|
||||||
|
break;
|
||||||
|
id = iter.next();
|
||||||
|
}
|
||||||
|
|
||||||
|
return ids;
|
||||||
|
`
|
||||||
|
});
|
||||||
|
|
||||||
|
if (rows.length)
|
||||||
|
callback(rows);
|
||||||
|
else
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
await db.select({
|
||||||
|
table: 'buc',
|
||||||
|
where: `
|
||||||
|
@delItem(${db.esc(iterName)});
|
||||||
|
return new Set();
|
||||||
|
`
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
async updateBuc(bookUrls) {
|
||||||
|
const db = this.db;
|
||||||
|
const now = Date.now();
|
||||||
|
|
||||||
|
const rows = await db.select({
|
||||||
|
table: 'buc',
|
||||||
|
map: `(r) => ({id: r.id})`,
|
||||||
|
where: `@@id(${db.esc(bookUrls)})`
|
||||||
|
});
|
||||||
|
|
||||||
|
const exists = new Set();
|
||||||
|
for (const row of rows) {
|
||||||
|
exists.add(row.id);
|
||||||
|
}
|
||||||
|
|
||||||
|
const toUpdateIds = [];
|
||||||
|
const toInsertRows = [];
|
||||||
|
for (let id of bookUrls) {
|
||||||
|
if (!id)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (id.length > 1000) {
|
||||||
|
id = id.substring(0, 1000);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (exists.has(id)) {
|
||||||
|
toUpdateIds.push(id);
|
||||||
|
} else {
|
||||||
|
toInsertRows.push({
|
||||||
|
id,
|
||||||
|
queryTime: now,
|
||||||
|
checkTime: 0, // 0 - never checked
|
||||||
|
etag: '',
|
||||||
|
modTime: '',
|
||||||
|
size: 0,
|
||||||
|
checkSum: '', //sha256
|
||||||
|
state: 0, // 0 - not processing, 1 - processing
|
||||||
|
error: '',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (toUpdateIds.length) {
|
||||||
|
await db.update({
|
||||||
|
table: 'buc',
|
||||||
|
mod: `(r) => r.queryTime = ${db.esc(now)}`,
|
||||||
|
where: `@@id(${db.esc(toUpdateIds)})`
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (toInsertRows.length) {
|
||||||
|
await db.insert({
|
||||||
|
table: 'buc',
|
||||||
|
ignore: true,
|
||||||
|
rows: toInsertRows,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async fillCheckQueue() {
|
||||||
|
const db = this.db;
|
||||||
|
|
||||||
|
while (1) {//eslint-disable-line
|
||||||
|
try {
|
||||||
|
let now = Date.now();
|
||||||
|
|
||||||
|
//чистка совсем устаревших
|
||||||
|
let rows = await db.select({
|
||||||
|
table: 'buc',
|
||||||
|
where: `@@dirtyIndexLR('queryTime', undefined, ${db.esc(now - this.cleanQueryInterval)})`
|
||||||
|
});
|
||||||
|
|
||||||
|
if (rows.length) {
|
||||||
|
const ids = rows.map((r) => r.id);
|
||||||
|
const res = await db.delete({
|
||||||
|
table: 'buc',
|
||||||
|
where: `@@id(${db.esc(ids)})`,
|
||||||
|
});
|
||||||
|
|
||||||
|
log(LM_WARN, `clean 'buc' table: deleted ${res.deleted}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
rows = await db.select({table: 'buc', count: true});
|
||||||
|
log(LM_WARN, `'buc' table size: ${rows[0].count}`);
|
||||||
|
|
||||||
|
now = Date.now();
|
||||||
|
//выборка кандидатов
|
||||||
|
rows = await db.select({
|
||||||
|
table: 'buc',
|
||||||
|
where: `
|
||||||
|
@@and(
|
||||||
|
@dirtyIndexLR('queryTime', ${db.esc(now - this.oldQueryInterval)}),
|
||||||
|
@dirtyIndexLR('checkTime', undefined, ${db.esc(now - this.checkingInterval)}),
|
||||||
|
@flag('notProcessing')
|
||||||
|
);
|
||||||
|
`
|
||||||
|
});
|
||||||
|
|
||||||
|
//формирование checkQueue
|
||||||
|
if (rows.length) {
|
||||||
|
const ids = [];
|
||||||
|
const rowsToPush = [];
|
||||||
|
|
||||||
|
//сначала выберем сколько надо
|
||||||
|
for (const row of rows) {
|
||||||
|
if (this.checkQueue.length + rowsToPush.length >= this.maxCheckQueueLength)
|
||||||
|
break;
|
||||||
|
|
||||||
|
rowsToPush.push(row);
|
||||||
|
ids.push(row.id);
|
||||||
|
}
|
||||||
|
|
||||||
|
//установим у них флаг "в обработке"
|
||||||
|
await db.update({
|
||||||
|
table: 'buc',
|
||||||
|
mod: `(r) => r.state = 1`,
|
||||||
|
where: `@@id(${db.esc(ids)})`
|
||||||
|
});
|
||||||
|
|
||||||
|
//пушим в очередь, после этого их обработает periodicCheck
|
||||||
|
for (const row of rowsToPush) {
|
||||||
|
this.checkQueue.push(row);
|
||||||
|
log(LM_INFO, ` add ${row.id}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
log(LM_WARN, `checkQueue: added ${ids.length} recs, total ${this.checkQueue.length}`);
|
||||||
|
}
|
||||||
|
} catch(e) {
|
||||||
|
log(LM_ERR, e.stack);
|
||||||
|
}
|
||||||
|
|
||||||
|
await utils.sleep(this.fillCheckQueuePeriod);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async periodicCheck() {
|
||||||
|
const db = this.db;
|
||||||
|
|
||||||
|
while (1) {//eslint-disable-line
|
||||||
|
try {
|
||||||
|
if (!this.checkQueue.length)
|
||||||
|
await utils.sleep(this.periodicCheckWait);
|
||||||
|
|
||||||
|
if (!this.checkQueue.length)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
const row = this.checkQueue.shift();
|
||||||
|
|
||||||
|
const url = new URL(row.id);
|
||||||
|
|
||||||
|
//только если обращались к тому же хосту не ранее sameHostCheckInterval миллисекунд назад
|
||||||
|
if (!this.hostChecking[url.hostname]) {
|
||||||
|
this.hostChecking[url.hostname] = true;
|
||||||
|
|
||||||
|
try {
|
||||||
|
let unchanged = true;
|
||||||
|
let hash = '';
|
||||||
|
|
||||||
|
const headers = await this.down.head(row.id);
|
||||||
|
|
||||||
|
const etag = headers['etag'] || '';
|
||||||
|
const modTime = headers['last-modified'] || '';
|
||||||
|
let size = parseInt(headers['content-length'], 10) || 0;
|
||||||
|
|
||||||
|
//log(row.id);
|
||||||
|
//log(`etag: ${etag}, modTime: ${modTime}, size: ${size}`)
|
||||||
|
|
||||||
|
if ((!etag || !row.etag || (etag !== row.etag))
|
||||||
|
&& (!modTime || !row.modTime || (modTime !== row.modTime))
|
||||||
|
&& (!size || !row.size || (size !== row.size))
|
||||||
|
) {
|
||||||
|
|
||||||
|
const downdata = await this.down.load(row.id);
|
||||||
|
|
||||||
|
size = downdata.length;
|
||||||
|
hash = await utils.getBufHash(downdata, 'sha256', 'hex');
|
||||||
|
unchanged = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
await db.update({
|
||||||
|
table: 'buc',
|
||||||
|
mod: `(r) => {
|
||||||
|
r.checkTime = ${db.esc(Date.now())};
|
||||||
|
r.etag = ${(unchanged ? 'r.etag' : db.esc(etag))};
|
||||||
|
r.modTime = ${(unchanged ? 'r.modTime' : db.esc(modTime))};
|
||||||
|
r.size = ${(unchanged ? 'r.size' : db.esc(size))};
|
||||||
|
r.checkSum = ${(unchanged ? 'r.checkSum' : db.esc(hash))};
|
||||||
|
r.state = 0;
|
||||||
|
r.error = '';
|
||||||
|
}`,
|
||||||
|
where: `@@id(${db.esc(row.id)})`
|
||||||
|
});
|
||||||
|
|
||||||
|
if (unchanged) {
|
||||||
|
log(`checked ${row.id} > unchanged`);
|
||||||
|
} else {
|
||||||
|
log(`checked ${row.id} > size ${size}`);
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
await db.update({
|
||||||
|
table: 'buc',
|
||||||
|
mod: `(r) => {
|
||||||
|
r.checkTime = ${db.esc(Date.now())};
|
||||||
|
r.state = 0;
|
||||||
|
r.error = ${db.esc(e.message)};
|
||||||
|
}`,
|
||||||
|
where: `@@id(${db.esc(row.id)})`
|
||||||
|
});
|
||||||
|
|
||||||
|
log(LM_ERR, `error ${row.id} > ${e.stack ? e.stack : e.message}`);
|
||||||
|
} finally {
|
||||||
|
(async() => {
|
||||||
|
await utils.sleep(this.sameHostCheckInterval);
|
||||||
|
this.hostChecking[url.hostname] = false;
|
||||||
|
})();
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
this.checkQueue.push(row);
|
||||||
|
}
|
||||||
|
} catch(e) {
|
||||||
|
log(LM_ERR, e.stack);
|
||||||
|
}
|
||||||
|
|
||||||
|
await utils.sleep(10);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async main() {
|
||||||
|
try {
|
||||||
|
//обнуляем все статусы
|
||||||
|
await this.db.update({table: 'buc', mod: `(r) => r.state = 0`});
|
||||||
|
|
||||||
|
this.fillCheckQueue();//no await
|
||||||
|
|
||||||
|
//10 потоков
|
||||||
|
for (let i = 0; i < 10; i++)
|
||||||
|
this.periodicCheck();//no await
|
||||||
|
|
||||||
|
log(`-------------------------`);
|
||||||
|
log(`BUC Server Worker started`);
|
||||||
|
log(`-------------------------`);
|
||||||
|
} catch (e) {
|
||||||
|
log(LM_FATAL, e.stack);
|
||||||
|
ayncExit.exit(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = BUCServer;
|
||||||
@@ -1,4 +1,7 @@
|
|||||||
const axios = require('axios');
|
const axios = require('axios');
|
||||||
|
const utils = require('./utils');
|
||||||
|
|
||||||
|
const userAgent = 'Mozilla/5.0 (X11; HasCodingOs 1.0; Linux x64) AppleWebKit/637.36 (KHTML, like Gecko) Chrome/70.0.3112.101 Safari/637.36 HasBrowser/5.0';
|
||||||
|
|
||||||
class FileDownloader {
|
class FileDownloader {
|
||||||
constructor(limitDownloadSize = 0) {
|
constructor(limitDownloadSize = 0) {
|
||||||
@@ -10,7 +13,8 @@ class FileDownloader {
|
|||||||
|
|
||||||
const options = {
|
const options = {
|
||||||
headers: {
|
headers: {
|
||||||
'user-agent': 'Mozilla/5.0 (X11; HasCodingOs 1.0; Linux x64) AppleWebKit/637.36 (KHTML, like Gecko) Chrome/70.0.3112.101 Safari/637.36 HasBrowser/5.0'
|
'user-agent': userAgent,
|
||||||
|
timeout: 300*1000,
|
||||||
},
|
},
|
||||||
responseType: 'stream',
|
responseType: 'stream',
|
||||||
};
|
};
|
||||||
@@ -23,7 +27,7 @@ class FileDownloader {
|
|||||||
estSize = res.headers['content-length'];
|
estSize = res.headers['content-length'];
|
||||||
}
|
}
|
||||||
|
|
||||||
if (estSize > this.limitDownloadSize) {
|
if (this.limitDownloadSize && estSize > this.limitDownloadSize) {
|
||||||
throw new Error('Файл слишком большой');
|
throw new Error('Файл слишком большой');
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -62,25 +66,54 @@ class FileDownloader {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
streamToBuffer(stream, progress) {
|
async head(url) {
|
||||||
|
const options = {
|
||||||
|
headers: {
|
||||||
|
'user-agent': userAgent,
|
||||||
|
timeout: 10*1000,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
const res = await axios.head(url, options);
|
||||||
|
return res.headers;
|
||||||
|
}
|
||||||
|
|
||||||
|
streamToBuffer(stream, progress, timeout = 30*1000) {
|
||||||
return new Promise((resolve, reject) => {
|
return new Promise((resolve, reject) => {
|
||||||
|
|
||||||
if (!progress)
|
if (!progress)
|
||||||
progress = () => {};
|
progress = () => {};
|
||||||
|
|
||||||
const _buf = [];
|
const _buf = [];
|
||||||
|
let resolved = false;
|
||||||
|
let timer = 0;
|
||||||
|
|
||||||
stream.on('data', (chunk) => {
|
stream.on('data', (chunk) => {
|
||||||
|
timer = 0;
|
||||||
_buf.push(chunk);
|
_buf.push(chunk);
|
||||||
progress(chunk);
|
progress(chunk);
|
||||||
});
|
});
|
||||||
stream.on('end', () => resolve(Buffer.concat(_buf)));
|
stream.on('end', () => {
|
||||||
|
resolved = true;
|
||||||
|
timer = timeout;
|
||||||
|
resolve(Buffer.concat(_buf));
|
||||||
|
});
|
||||||
stream.on('error', (err) => {
|
stream.on('error', (err) => {
|
||||||
reject(err);
|
reject(err);
|
||||||
});
|
});
|
||||||
stream.on('aborted', () => {
|
stream.on('aborted', () => {
|
||||||
reject(new Error('aborted'));
|
reject(new Error('aborted'));
|
||||||
});
|
});
|
||||||
|
|
||||||
|
//бодяга с timer и timeout, чтобы гарантировать отсутствие зависания по каким-либо причинам
|
||||||
|
(async() => {
|
||||||
|
while (timer < timeout) {
|
||||||
|
await utils.sleep(1000);
|
||||||
|
timer += 1000;
|
||||||
|
}
|
||||||
|
if (!resolved)
|
||||||
|
reject(new Error('FileDownloader: timed out'))
|
||||||
|
})();
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,12 +6,16 @@ const WorkerState = require('../WorkerState');//singleton
|
|||||||
const FileDownloader = require('../FileDownloader');
|
const FileDownloader = require('../FileDownloader');
|
||||||
const FileDecompressor = require('../FileDecompressor');
|
const FileDecompressor = require('../FileDecompressor');
|
||||||
const BookConverter = require('./BookConverter');
|
const BookConverter = require('./BookConverter');
|
||||||
const RemoteWebDavStorage = require('../RemoteWebDavStorage');
|
const RemoteStorage = require('../RemoteStorage');
|
||||||
|
const JembaConnManager = require('../../db/JembaConnManager');//singleton
|
||||||
|
const ayncExit = new (require('../AsyncExit'))();
|
||||||
|
|
||||||
const utils = require('../utils');
|
const utils = require('../utils');
|
||||||
const log = new (require('../AppLogger'))().log;//singleton
|
const log = new (require('../AppLogger'))().log;//singleton
|
||||||
|
|
||||||
const cleanDirPeriod = 60*60*1000;//1 раз в час
|
const cleanDirPeriod = 60*60*1000;//каждый час
|
||||||
|
const remoteSendPeriod = 119*1000;//примерно раз 2 минуты
|
||||||
|
|
||||||
const queue = new LimitedQueue(5, 100, 2*60*1000 + 15000);//2 минуты ожидание подвижек
|
const queue = new LimitedQueue(5, 100, 2*60*1000 + 15000);//2 минуты ожидание подвижек
|
||||||
|
|
||||||
let instance = null;
|
let instance = null;
|
||||||
@@ -33,15 +37,37 @@ class ReaderWorker {
|
|||||||
this.decomp = new FileDecompressor(3*config.maxUploadFileSize);
|
this.decomp = new FileDecompressor(3*config.maxUploadFileSize);
|
||||||
this.bookConverter = new BookConverter(this.config);
|
this.bookConverter = new BookConverter(this.config);
|
||||||
|
|
||||||
this.remoteWebDavStorage = false;
|
this.connManager = new JembaConnManager();
|
||||||
if (config.remoteWebDavStorage) {
|
this.appDb = this.connManager.db['app'];
|
||||||
this.remoteWebDavStorage = new RemoteWebDavStorage(
|
|
||||||
Object.assign({maxContentLength: 3*config.maxUploadFileSize}, config.remoteWebDavStorage)
|
this.remoteStorage = false;
|
||||||
|
if (config.remoteStorage) {
|
||||||
|
this.remoteStorage = new RemoteStorage(
|
||||||
|
Object.assign({maxContentLength: 3*config.maxUploadFileSize}, config.remoteStorage)
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
this.periodicCleanDir(this.config.tempPublicDir, this.config.maxTempPublicDirSize, cleanDirPeriod);
|
this.dirConfigArr = [
|
||||||
this.periodicCleanDir(this.config.uploadDir, this.config.maxUploadPublicDirSize, cleanDirPeriod);
|
{
|
||||||
|
dir: this.config.tempPublicDir,
|
||||||
|
remoteDir: '/tmp',
|
||||||
|
maxSize: this.config.maxTempPublicDirSize,
|
||||||
|
moveToRemote: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
dir: this.config.uploadDir,
|
||||||
|
remoteDir: '/upload',
|
||||||
|
maxSize: this.config.maxUploadPublicDirSize,
|
||||||
|
moveToRemote: true,
|
||||||
|
}
|
||||||
|
];
|
||||||
|
//преобразуем в объект для большего удобства
|
||||||
|
this.dirConfig = {};
|
||||||
|
for (const configRec of this.dirConfigArr)
|
||||||
|
this.dirConfig[configRec.remoteDir] = configRec;
|
||||||
|
|
||||||
|
this.remoteFilesToSend = [];
|
||||||
|
this.periodicCleanDir();//no await
|
||||||
|
|
||||||
instance = this;
|
instance = this;
|
||||||
}
|
}
|
||||||
@@ -54,7 +80,6 @@ class ReaderWorker {
|
|||||||
let decompDir = '';
|
let decompDir = '';
|
||||||
let downloadedFilename = '';
|
let downloadedFilename = '';
|
||||||
let isUploaded = false;
|
let isUploaded = false;
|
||||||
let isRestored = false;
|
|
||||||
let convertFilename = '';
|
let convertFilename = '';
|
||||||
|
|
||||||
const overLoadMes = 'Слишком большая очередь загрузки. Пожалуйста, попробуйте позже.';
|
const overLoadMes = 'Слишком большая очередь загрузки. Пожалуйста, попробуйте позже.';
|
||||||
@@ -80,6 +105,7 @@ class ReaderWorker {
|
|||||||
const tempFilename2 = utils.randomHexString(30);
|
const tempFilename2 = utils.randomHexString(30);
|
||||||
const decompDirname = utils.randomHexString(30);
|
const decompDirname = utils.randomHexString(30);
|
||||||
|
|
||||||
|
let downloadSize = -1;
|
||||||
//download or use uploaded
|
//download or use uploaded
|
||||||
if (url.indexOf('disk://') != 0) {//download
|
if (url.indexOf('disk://') != 0) {//download
|
||||||
const downdata = await this.down.load(url, (progress) => {
|
const downdata = await this.down.load(url, (progress) => {
|
||||||
@@ -87,6 +113,8 @@ class ReaderWorker {
|
|||||||
}, q.abort);
|
}, q.abort);
|
||||||
|
|
||||||
downloadedFilename = `${this.config.tempDownloadDir}/${tempFilename}`;
|
downloadedFilename = `${this.config.tempDownloadDir}/${tempFilename}`;
|
||||||
|
|
||||||
|
downloadSize = downdata.length;
|
||||||
await fs.writeFile(downloadedFilename, downdata);
|
await fs.writeFile(downloadedFilename, downdata);
|
||||||
} else {//uploaded file
|
} else {//uploaded file
|
||||||
const fileHash = url.substr(7);
|
const fileHash = url.substr(7);
|
||||||
@@ -94,8 +122,7 @@ class ReaderWorker {
|
|||||||
if (!await fs.pathExists(downloadedFilename)) {
|
if (!await fs.pathExists(downloadedFilename)) {
|
||||||
//если удалено из upload, попробуем восстановить из удаленного хранилища
|
//если удалено из upload, попробуем восстановить из удаленного хранилища
|
||||||
try {
|
try {
|
||||||
downloadedFilename = await this.restoreRemoteFile(fileHash);
|
await this.restoreRemoteFile(fileHash, '/upload');
|
||||||
isRestored = true;
|
|
||||||
} catch(e) {
|
} catch(e) {
|
||||||
throw new Error('Файл не найден на сервере (возможно был удален как устаревший). Пожалуйста, загрузите файл с диска на сервер заново.');
|
throw new Error('Файл не найден на сервере (возможно был удален как устаревший). Пожалуйста, загрузите файл с диска на сервер заново.');
|
||||||
}
|
}
|
||||||
@@ -142,34 +169,19 @@ class ReaderWorker {
|
|||||||
|
|
||||||
//finish
|
//finish
|
||||||
const finishFilename = path.basename(compFilename);
|
const finishFilename = path.basename(compFilename);
|
||||||
wState.finish({path: `/tmp/${finishFilename}`, size: stat.size});
|
|
||||||
|
|
||||||
//лениво сохраним compFilename в удаленном хранилище
|
const result = {path: `/tmp/${finishFilename}`, size: stat.size};
|
||||||
if (this.remoteWebDavStorage) {
|
if (downloadSize >= 0)
|
||||||
(async() => {
|
result.downloadSize = downloadSize;
|
||||||
await utils.sleep(20*1000);
|
|
||||||
try {
|
|
||||||
//log(`remoteWebDavStorage.putFile ${path.basename(compFilename)}`);
|
|
||||||
await this.remoteWebDavStorage.putFile(compFilename);
|
|
||||||
} catch (e) {
|
|
||||||
log(LM_ERR, e.stack);
|
|
||||||
}
|
|
||||||
})();
|
|
||||||
}
|
|
||||||
|
|
||||||
//лениво сохраним downloadedFilename в tmp и в удаленном хранилище в случае isUploaded
|
wState.finish(result);
|
||||||
if (this.remoteWebDavStorage && isUploaded && !isRestored) {
|
|
||||||
(async() => {
|
//асинхронно через 30 сек добавим в очередь на отправку
|
||||||
await utils.sleep(30*1000);
|
//т.к. gzipFileIfNotExists может переупаковать файл
|
||||||
try {
|
(async() => {
|
||||||
//сжимаем файл в tmp, если там уже нет с тем же именем-sha256
|
await utils.sleep(30*1000);
|
||||||
const compDownloadedFilename = await this.decomp.gzipFileIfNotExists(downloadedFilename, this.config.tempPublicDir, true);
|
this.pushRemoteSend(compFilename, '/tmp');
|
||||||
await this.remoteWebDavStorage.putFile(compDownloadedFilename);
|
})();
|
||||||
} catch (e) {
|
|
||||||
log(LM_ERR, e.stack);
|
|
||||||
}
|
|
||||||
})();
|
|
||||||
}
|
|
||||||
|
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
log(LM_ERR, e.stack);
|
log(LM_ERR, e.stack);
|
||||||
@@ -211,6 +223,7 @@ class ReaderWorker {
|
|||||||
|
|
||||||
if (!await fs.pathExists(outFilename)) {
|
if (!await fs.pathExists(outFilename)) {
|
||||||
await fs.move(file.path, outFilename);
|
await fs.move(file.path, outFilename);
|
||||||
|
this.pushRemoteSend(outFilename, '/upload');
|
||||||
} else {
|
} else {
|
||||||
await utils.touchFile(outFilename);
|
await utils.touchFile(outFilename);
|
||||||
await fs.remove(file.path);
|
await fs.remove(file.path);
|
||||||
@@ -219,14 +232,42 @@ class ReaderWorker {
|
|||||||
return `disk://${hash}`;
|
return `disk://${hash}`;
|
||||||
}
|
}
|
||||||
|
|
||||||
async restoreRemoteFile(filename) {
|
async saveFileBuf(buf) {
|
||||||
|
const hash = await utils.getBufHash(buf, 'sha256', 'hex');
|
||||||
|
const outFilename = `${this.config.uploadDir}/${hash}`;
|
||||||
|
|
||||||
|
if (!await fs.pathExists(outFilename)) {
|
||||||
|
await fs.writeFile(outFilename, buf);
|
||||||
|
this.pushRemoteSend(outFilename, '/upload');
|
||||||
|
} else {
|
||||||
|
await utils.touchFile(outFilename);
|
||||||
|
}
|
||||||
|
|
||||||
|
return `disk://${hash}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
async uploadFileTouch(url) {
|
||||||
|
const outFilename = `${this.config.uploadDir}/${url.replace('disk://', '')}`;
|
||||||
|
|
||||||
|
await utils.touchFile(outFilename);
|
||||||
|
|
||||||
|
return url;
|
||||||
|
}
|
||||||
|
|
||||||
|
async restoreRemoteFile(filename, remoteDir) {
|
||||||
|
let targetDir = '';
|
||||||
|
if (this.dirConfig[remoteDir])
|
||||||
|
targetDir = this.dirConfig[remoteDir].dir;
|
||||||
|
else
|
||||||
|
throw new Error(`restoreRemoteFile: unknown remoteDir value (${remoteDir})`);
|
||||||
|
|
||||||
const basename = path.basename(filename);
|
const basename = path.basename(filename);
|
||||||
const targetName = `${this.config.tempPublicDir}/${basename}`;
|
const targetName = `${targetDir}/${basename}`;
|
||||||
|
|
||||||
if (!await fs.pathExists(targetName)) {
|
if (!await fs.pathExists(targetName)) {
|
||||||
let found = false;
|
let found = false;
|
||||||
if (this.remoteWebDavStorage) {
|
if (this.remoteStorage) {
|
||||||
found = await this.remoteWebDavStorage.getFileSuccess(targetName);
|
found = await this.remoteStorage.getFileSuccess(targetName, remoteDir);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!found) {
|
if (!found) {
|
||||||
@@ -237,83 +278,170 @@ class ReaderWorker {
|
|||||||
return targetName;
|
return targetName;
|
||||||
}
|
}
|
||||||
|
|
||||||
restoreCachedFile(filename) {
|
pushRemoteSend(fileName, remoteDir) {
|
||||||
const workerId = this.workerState.generateWorkerId();
|
if (this.remoteStorage
|
||||||
const wState = this.workerState.getControl(workerId);
|
&& this.dirConfig[remoteDir]
|
||||||
wState.set({state: 'start'});
|
&& this.dirConfig[remoteDir].moveToRemote) {
|
||||||
|
this.remoteFilesToSend.push({fileName, remoteDir});
|
||||||
(async() => {
|
|
||||||
try {
|
|
||||||
wState.set({state: 'download', step: 1, totalSteps: 1, path: filename, progress: 0});
|
|
||||||
|
|
||||||
const targetName = await this.restoreRemoteFile(filename);
|
|
||||||
const stat = await fs.stat(targetName);
|
|
||||||
|
|
||||||
const basename = path.basename(filename);
|
|
||||||
wState.finish({path: `/tmp/${basename}`, size: stat.size, progress: 100});
|
|
||||||
} catch (e) {
|
|
||||||
if (e.message.indexOf('404') < 0)
|
|
||||||
log(LM_ERR, e.stack);
|
|
||||||
wState.set({state: 'error', error: e.message});
|
|
||||||
}
|
|
||||||
})();
|
|
||||||
|
|
||||||
return workerId;
|
|
||||||
}
|
|
||||||
|
|
||||||
async periodicCleanDir(dir, maxSize, timeout) {
|
|
||||||
try {
|
|
||||||
const list = await fs.readdir(dir);
|
|
||||||
|
|
||||||
let size = 0;
|
|
||||||
let files = [];
|
|
||||||
for (const name of list) {
|
|
||||||
const stat = await fs.stat(`${dir}/${name}`);
|
|
||||||
if (!stat.isDirectory()) {
|
|
||||||
size += stat.size;
|
|
||||||
files.push({name, stat});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
log(`clean dir ${dir}, maxSize=${maxSize}, found ${files.length} files, total size=${size}`);
|
|
||||||
|
|
||||||
files.sort((a, b) => a.stat.mtimeMs - b.stat.mtimeMs);
|
|
||||||
|
|
||||||
let i = 0;
|
|
||||||
let j = 0;
|
|
||||||
while (i < files.length && size > maxSize) {
|
|
||||||
const file = files[i];
|
|
||||||
const oldFile = `${dir}/${file.name}`;
|
|
||||||
|
|
||||||
let remoteSuccess = true;
|
|
||||||
//отправляем только this.config.tempPublicDir
|
|
||||||
if (this.remoteWebDavStorage && dir === this.config.tempPublicDir) {
|
|
||||||
remoteSuccess = false;
|
|
||||||
try {
|
|
||||||
//log(`remoteWebDavStorage.putFile ${path.basename(oldFile)}`);
|
|
||||||
await this.remoteWebDavStorage.putFile(oldFile);
|
|
||||||
remoteSuccess = true;
|
|
||||||
} catch (e) {
|
|
||||||
log(LM_ERR, e.stack);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
//реально удаляем только если сохранили в хранилище
|
|
||||||
if (remoteSuccess || size > maxSize*1.2) {
|
|
||||||
await fs.remove(oldFile);
|
|
||||||
j++;
|
|
||||||
}
|
|
||||||
|
|
||||||
size -= file.stat.size;
|
|
||||||
i++;
|
|
||||||
}
|
|
||||||
log(`removed ${j} files`);
|
|
||||||
} catch(e) {
|
|
||||||
log(LM_ERR, e.stack);
|
|
||||||
} finally {
|
|
||||||
setTimeout(() => {
|
|
||||||
this.periodicCleanDir(dir, maxSize, timeout);
|
|
||||||
}, timeout);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async remoteSendFile(sendFileRec) {
|
||||||
|
const {fileName, remoteDir} = sendFileRec;
|
||||||
|
const sent = this.remoteSent;
|
||||||
|
|
||||||
|
if (!fileName || sent[fileName])
|
||||||
|
return;
|
||||||
|
|
||||||
|
log(`remoteSendFile ${remoteDir}/${path.basename(fileName)}`);
|
||||||
|
|
||||||
|
//отправляем в remoteStorage
|
||||||
|
await this.remoteStorage.putFile(fileName, remoteDir);
|
||||||
|
|
||||||
|
sent[fileName] = true;
|
||||||
|
await this.appDb.insert({table: 'remote_sent', ignore: true, rows: [{id: fileName, remoteDir}]});
|
||||||
|
}
|
||||||
|
|
||||||
|
async remoteSendAll() {
|
||||||
|
if (!this.remoteStorage)
|
||||||
|
return;
|
||||||
|
|
||||||
|
const newSendQueue = [];
|
||||||
|
while (this.remoteFilesToSend.length) {
|
||||||
|
const sendFileRec = this.remoteFilesToSend.shift();
|
||||||
|
|
||||||
|
if (sendFileRec.remoteDir
|
||||||
|
&& this.dirConfig[sendFileRec.remoteDir]
|
||||||
|
&& this.dirConfig[sendFileRec.remoteDir].moveToRemote) {
|
||||||
|
|
||||||
|
try {
|
||||||
|
await this.remoteSendFile(sendFileRec);
|
||||||
|
} catch (e) {
|
||||||
|
newSendQueue.push(sendFileRec)
|
||||||
|
log(LM_ERR, e.stack);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
this.remoteFilesToSend = newSendQueue;
|
||||||
|
}
|
||||||
|
|
||||||
|
async cleanDir(config) {
|
||||||
|
const {dir, remoteDir, maxSize, moveToRemote} = config;
|
||||||
|
const sent = this.remoteSent;
|
||||||
|
|
||||||
|
const list = await fs.readdir(dir);
|
||||||
|
|
||||||
|
let size = 0;
|
||||||
|
let files = [];
|
||||||
|
for (const filename of list) {
|
||||||
|
const filePath = `${dir}/${filename}`;
|
||||||
|
const stat = await fs.stat(filePath);
|
||||||
|
if (!stat.isDirectory()) {
|
||||||
|
size += stat.size;
|
||||||
|
files.push({name: filePath, stat});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
log(LM_WARN, `clean dir ${dir}, maxSize=${maxSize}, found ${files.length} files, total size=${size}`);
|
||||||
|
|
||||||
|
files.sort((a, b) => a.stat.mtimeMs - b.stat.mtimeMs);
|
||||||
|
|
||||||
|
//удаленное хранилище
|
||||||
|
if (moveToRemote && this.remoteStorage) {
|
||||||
|
const foundFiles = new Set();
|
||||||
|
for (const file of files) {
|
||||||
|
foundFiles.add(file.name);
|
||||||
|
|
||||||
|
//отсылаем на всякий случай перед удалением, если вдруг remoteSendAll не справился
|
||||||
|
try {
|
||||||
|
await this.remoteSendFile({fileName: file.name, remoteDir});
|
||||||
|
} catch (e) {
|
||||||
|
log(LM_ERR, e.stack);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//почистим remoteSent и БД
|
||||||
|
//несколько неоптимально, таскает все записи из таблицы
|
||||||
|
const rows = await this.appDb.select({table: 'remote_sent'});
|
||||||
|
for (const row of rows) {
|
||||||
|
if ((row.remoteDir === remoteDir && !foundFiles.has(row.id))
|
||||||
|
|| !this.dirConfig[row.remoteDir]) {
|
||||||
|
delete sent[row.id];
|
||||||
|
await this.appDb.delete({table: 'remote_sent', where: `@@id(${this.appDb.esc(row.id)})`});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let i = 0;
|
||||||
|
let j = 0;
|
||||||
|
while (i < files.length && size > maxSize) {
|
||||||
|
const file = files[i];
|
||||||
|
const oldFile = file.name;
|
||||||
|
|
||||||
|
//реально удаляем только если сохранили в хранилище или размер dir увеличен в 1.5 раза
|
||||||
|
if (!(moveToRemote && this.remoteStorage)
|
||||||
|
|| (moveToRemote && this.remoteStorage && sent[oldFile])
|
||||||
|
|| size > maxSize*1.5) {
|
||||||
|
await fs.remove(oldFile);
|
||||||
|
j++;
|
||||||
|
}
|
||||||
|
|
||||||
|
size -= file.stat.size;
|
||||||
|
i++;
|
||||||
|
}
|
||||||
|
|
||||||
|
log(LM_WARN, `removed ${j} files`);
|
||||||
|
}
|
||||||
|
|
||||||
|
async periodicCleanDir() {
|
||||||
|
try {
|
||||||
|
if (!this.remoteSent)
|
||||||
|
this.remoteSent = {};
|
||||||
|
|
||||||
|
//инициализация this.remoteSent
|
||||||
|
if (this.remoteStorage) {
|
||||||
|
const rows = await this.appDb.select({table: 'remote_sent'});
|
||||||
|
for (const row of rows) {
|
||||||
|
this.remoteSent[row.id] = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let lastCleanDirTime = 0;
|
||||||
|
let lastRemoteSendTime = 0;
|
||||||
|
while (1) {// eslint-disable-line no-constant-condition
|
||||||
|
//отсылка в удаленное хранилище
|
||||||
|
if (Date.now() - lastRemoteSendTime >= remoteSendPeriod) {
|
||||||
|
try {
|
||||||
|
await this.remoteSendAll();
|
||||||
|
} catch(e) {
|
||||||
|
log(LM_ERR, e.stack);
|
||||||
|
}
|
||||||
|
|
||||||
|
lastRemoteSendTime = Date.now();
|
||||||
|
}
|
||||||
|
|
||||||
|
//чистка папок
|
||||||
|
if (Date.now() - lastCleanDirTime >= cleanDirPeriod) {
|
||||||
|
for (const config of Object.values(this.dirConfig)) {
|
||||||
|
try {
|
||||||
|
await this.cleanDir(config);
|
||||||
|
} catch(e) {
|
||||||
|
log(LM_ERR, e.stack);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
lastCleanDirTime = Date.now();
|
||||||
|
}
|
||||||
|
|
||||||
|
await utils.sleep(60*1000);//интервал проверки 1 минута
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
log(LM_FATAL, e.message);
|
||||||
|
ayncExit.exit(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
module.exports = ReaderWorker;
|
module.exports = ReaderWorker;
|
||||||
98
server/core/RemoteStorage.js
Normal file
98
server/core/RemoteStorage.js
Normal file
@@ -0,0 +1,98 @@
|
|||||||
|
const fs = require('fs-extra');
|
||||||
|
const path = require('path');
|
||||||
|
|
||||||
|
const WebSocketConnection = require('./WebSocketConnection');
|
||||||
|
|
||||||
|
class RemoteStorage {
|
||||||
|
constructor(config) {
|
||||||
|
this.config = Object.assign({}, config);
|
||||||
|
this.config.maxContentLength = this.config.maxContentLength || 10*1024*1024;
|
||||||
|
|
||||||
|
this.accessToken = this.config.accessToken;
|
||||||
|
|
||||||
|
this.wsc = new WebSocketConnection(config.url, 10, 30, {rejectUnauthorized: false});
|
||||||
|
}
|
||||||
|
|
||||||
|
async wsRequest(query) {
|
||||||
|
const response = await this.wsc.message(
|
||||||
|
await this.wsc.send(Object.assign({accessToken: this.accessToken}, query), 600),
|
||||||
|
600
|
||||||
|
);
|
||||||
|
if (response.error)
|
||||||
|
throw new Error(response.error);
|
||||||
|
return response;
|
||||||
|
}
|
||||||
|
|
||||||
|
async wsStat(fileName) {
|
||||||
|
return await this.wsRequest({action: 'get-stat', fileName});
|
||||||
|
}
|
||||||
|
|
||||||
|
async wsGetFile(fileName) {
|
||||||
|
return this.wsRequest({action: 'get-file', fileName});
|
||||||
|
}
|
||||||
|
|
||||||
|
async wsPutFile(fileName, data) {//data base64 encoded string
|
||||||
|
return this.wsRequest({action: 'put-file', fileName, data});
|
||||||
|
}
|
||||||
|
|
||||||
|
async wsDelFile(fileName) {
|
||||||
|
return this.wsRequest({action: 'del-file', fileName});
|
||||||
|
}
|
||||||
|
|
||||||
|
makeRemoteFileName(fileName, dir = '') {
|
||||||
|
const base = path.basename(fileName);
|
||||||
|
if (base.length > 3) {
|
||||||
|
return `${dir}/${base.substr(0, 3)}/${base}`;
|
||||||
|
} else {
|
||||||
|
return `${dir}/${base}`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async putFile(fileName, dir = '') {
|
||||||
|
if (!await fs.pathExists(fileName)) {
|
||||||
|
throw new Error(`File not found: ${fileName}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
const remoteFilename = this.makeRemoteFileName(fileName, dir);
|
||||||
|
|
||||||
|
try {
|
||||||
|
const localStat = await fs.stat(fileName);
|
||||||
|
let remoteStat = await this.wsStat(remoteFilename);
|
||||||
|
remoteStat = remoteStat.stat;
|
||||||
|
|
||||||
|
if (remoteStat.isFile && localStat.size == remoteStat.size) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
await this.wsDelFile(remoteFilename);
|
||||||
|
} catch (e) {
|
||||||
|
//
|
||||||
|
}
|
||||||
|
|
||||||
|
const data = await fs.readFile(fileName, 'base64');
|
||||||
|
await this.wsPutFile(remoteFilename, data);
|
||||||
|
}
|
||||||
|
|
||||||
|
async getFile(fileName, dir = '') {
|
||||||
|
if (await fs.pathExists(fileName)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const remoteFilename = this.makeRemoteFileName(fileName, dir);
|
||||||
|
|
||||||
|
const response = await this.wsGetFile(remoteFilename);
|
||||||
|
await fs.writeFile(fileName, response.data, 'base64');
|
||||||
|
}
|
||||||
|
|
||||||
|
async getFileSuccess(filename, dir = '') {
|
||||||
|
try {
|
||||||
|
await this.getFile(filename, dir);
|
||||||
|
return true;
|
||||||
|
} catch (e) {
|
||||||
|
//
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = RemoteStorage;
|
||||||
@@ -46,16 +46,16 @@ class RemoteWebDavStorage {
|
|||||||
return await this.wdc.createDirectory(dirname);
|
return await this.wdc.createDirectory(dirname);
|
||||||
}
|
}
|
||||||
|
|
||||||
async putFile(filename) {
|
async putFile(filename, dir = '') {
|
||||||
if (!await fs.pathExists(filename)) {
|
if (!await fs.pathExists(filename)) {
|
||||||
throw new Error(`File not found: ${filename}`);
|
throw new Error(`File not found: ${filename}`);
|
||||||
}
|
}
|
||||||
|
|
||||||
const base = path.basename(filename);
|
const base = path.basename(filename);
|
||||||
let remoteFilename = `/${base}`;
|
let remoteFilename = `${dir}/${base}`;
|
||||||
|
|
||||||
if (base.length > 3) {
|
if (base.length > 3) {
|
||||||
const remoteDir = `/${base.substr(0, 3)}`;
|
const remoteDir = `${dir}/${base.substr(0, 3)}`;
|
||||||
try {
|
try {
|
||||||
await this.mkdir(remoteDir);
|
await this.mkdir(remoteDir);
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
@@ -79,24 +79,24 @@ class RemoteWebDavStorage {
|
|||||||
await this.writeFile(remoteFilename, data);
|
await this.writeFile(remoteFilename, data);
|
||||||
}
|
}
|
||||||
|
|
||||||
async getFile(filename) {
|
async getFile(filename, dir = '') {
|
||||||
if (await fs.pathExists(filename)) {
|
if (await fs.pathExists(filename)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const base = path.basename(filename);
|
const base = path.basename(filename);
|
||||||
let remoteFilename = `/${base}`;
|
let remoteFilename = `${dir}/${base}`;
|
||||||
if (base.length > 3) {
|
if (base.length > 3) {
|
||||||
remoteFilename = `/${base.substr(0, 3)}/${base}`;
|
remoteFilename = `${dir}/${base.substr(0, 3)}/${base}`;
|
||||||
}
|
}
|
||||||
|
|
||||||
const data = await this.readFile(remoteFilename);
|
const data = await this.readFile(remoteFilename);
|
||||||
await fs.writeFile(filename, data);
|
await fs.writeFile(filename, data);
|
||||||
}
|
}
|
||||||
|
|
||||||
async getFileSuccess(filename) {
|
async getFileSuccess(filename, dir = '') {
|
||||||
try {
|
try {
|
||||||
await this.getFile(filename);
|
await this.getFile(filename, dir);
|
||||||
return true;
|
return true;
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
//
|
//
|
||||||
|
|||||||
@@ -8,10 +8,13 @@ const cleanPeriod = 5*1000;//5 секунд
|
|||||||
|
|
||||||
class WebSocketConnection {
|
class WebSocketConnection {
|
||||||
//messageLifeTime в секундах (проверка каждый cleanPeriod интервал)
|
//messageLifeTime в секундах (проверка каждый cleanPeriod интервал)
|
||||||
constructor(url, openTimeoutSecs = 10, messageLifeTimeSecs = 30) {
|
constructor(url, openTimeoutSecs = 10, messageLifeTimeSecs = 30, webSocketOptions = {}) {
|
||||||
this.WebSocket = (isBrowser ? WebSocket : require('ws'));
|
this.WebSocket = (isBrowser ? WebSocket : require('ws'));
|
||||||
this.url = url;
|
this.url = url;
|
||||||
|
this.webSocketOptions = webSocketOptions;
|
||||||
|
|
||||||
this.ws = null;
|
this.ws = null;
|
||||||
|
|
||||||
this.listeners = [];
|
this.listeners = [];
|
||||||
this.messageQueue = [];
|
this.messageQueue = [];
|
||||||
this.messageLifeTime = messageLifeTimeSecs*1000;
|
this.messageLifeTime = messageLifeTimeSecs*1000;
|
||||||
@@ -91,10 +94,10 @@ class WebSocketConnection {
|
|||||||
const url = this.url || `${protocol}//${window.location.host}/ws`;
|
const url = this.url || `${protocol}//${window.location.host}/ws`;
|
||||||
this.ws = new this.WebSocket(url);
|
this.ws = new this.WebSocket(url);
|
||||||
} else {
|
} else {
|
||||||
this.ws = new this.WebSocket(this.url);
|
this.ws = new this.WebSocket(this.url, this.webSocketOptions);
|
||||||
}
|
}
|
||||||
|
|
||||||
const onopen = (e) => {
|
const onopen = () => {
|
||||||
this.connecting = false;
|
this.connecting = false;
|
||||||
resolve(this.ws);
|
resolve(this.ws);
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -25,7 +25,7 @@ class WorkerState {
|
|||||||
return {
|
return {
|
||||||
set: state => this.setState(workerId, state),
|
set: state => this.setState(workerId, state),
|
||||||
finish: state => this.finishState(workerId, state),
|
finish: state => this.finishState(workerId, state),
|
||||||
get: workerId => this.getState(workerId),
|
get: () => this.getState(workerId),
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -34,6 +34,12 @@ function getFileHash(filename, hashName, enc) {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function getBufHash(buf, hashName, enc) {
|
||||||
|
const hash = crypto.createHash(hashName);
|
||||||
|
hash.update(buf);
|
||||||
|
return hash.digest(enc);
|
||||||
|
}
|
||||||
|
|
||||||
function sleep(ms) {
|
function sleep(ms) {
|
||||||
return new Promise(resolve => setTimeout(resolve, ms));
|
return new Promise(resolve => setTimeout(resolve, ms));
|
||||||
}
|
}
|
||||||
@@ -129,6 +135,7 @@ module.exports = {
|
|||||||
fromBase36,
|
fromBase36,
|
||||||
bufferRemoveZeroes,
|
bufferRemoveZeroes,
|
||||||
getFileHash,
|
getFileHash,
|
||||||
|
getBufHash,
|
||||||
sleep,
|
sleep,
|
||||||
toUnixTime,
|
toUnixTime,
|
||||||
randomHexString,
|
randomHexString,
|
||||||
|
|||||||
@@ -1,61 +0,0 @@
|
|||||||
//TODO: удалить модуль в 2023г
|
|
||||||
const fs = require('fs-extra');
|
|
||||||
|
|
||||||
const SqliteConnectionPool = require('./SqliteConnectionPool');
|
|
||||||
const log = new (require('../core/AppLogger'))().log;//singleton
|
|
||||||
|
|
||||||
const migrations = {
|
|
||||||
'app': require('./migrations/app'),
|
|
||||||
'readerStorage': require('./migrations/readerStorage'),
|
|
||||||
};
|
|
||||||
|
|
||||||
let instance = null;
|
|
||||||
|
|
||||||
//singleton
|
|
||||||
class ConnManager {
|
|
||||||
constructor() {
|
|
||||||
if (!instance) {
|
|
||||||
this.inited = false;
|
|
||||||
|
|
||||||
instance = this;
|
|
||||||
}
|
|
||||||
|
|
||||||
return instance;
|
|
||||||
}
|
|
||||||
|
|
||||||
async init(config) {
|
|
||||||
this.config = config;
|
|
||||||
this._pool = {};
|
|
||||||
|
|
||||||
const force = null;//(config.branch == 'development' ? 'last' : null);
|
|
||||||
|
|
||||||
for (const poolConfig of this.config.db) {
|
|
||||||
const dbFileName = this.config.dataDir + '/' + poolConfig.fileName;
|
|
||||||
|
|
||||||
//бэкап
|
|
||||||
if (!poolConfig.noBak && await fs.pathExists(dbFileName))
|
|
||||||
await fs.copy(dbFileName, `${dbFileName}.bak`);
|
|
||||||
|
|
||||||
const connPool = new SqliteConnectionPool();
|
|
||||||
await connPool.open(poolConfig, dbFileName);
|
|
||||||
|
|
||||||
log(`Opened database "${poolConfig.poolName}"`);
|
|
||||||
//миграции
|
|
||||||
const migs = migrations[poolConfig.poolName];
|
|
||||||
if (migs && migs.data.length) {
|
|
||||||
const applied = await connPool.migrate(migs.data, migs.table, force);
|
|
||||||
if (applied.length)
|
|
||||||
log(`${applied.length} migrations applied to "${poolConfig.poolName}"`);
|
|
||||||
}
|
|
||||||
|
|
||||||
this._pool[poolConfig.poolName] = connPool;
|
|
||||||
}
|
|
||||||
this.inited = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
get pool() {
|
|
||||||
return this._pool;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
module.exports = ConnManager;
|
|
||||||
@@ -1,42 +0,0 @@
|
|||||||
//TODO: удалить модуль в 2023г
|
|
||||||
const fs = require('fs-extra');
|
|
||||||
const log = new (require('../core/AppLogger'))().log;//singleton
|
|
||||||
|
|
||||||
class Converter {
|
|
||||||
async run(config) {
|
|
||||||
log('Converter start');
|
|
||||||
|
|
||||||
try {
|
|
||||||
const connManager = new (require('./ConnManager'))();//singleton
|
|
||||||
const storagePool = connManager.pool.readerStorage;
|
|
||||||
|
|
||||||
const jembaConnManager = new (require('./JembaConnManager'))();//singleton
|
|
||||||
const db = jembaConnManager.db['reader-storage'];
|
|
||||||
|
|
||||||
const srcDbPath = `${config.dataDir}/reader-storage.sqlite`;
|
|
||||||
if (!await fs.pathExists(srcDbPath)) {
|
|
||||||
log(LM_WARN, ' Source DB does not exist, nothing to do');
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const rows = await db.select({table: 'storage', count: true});
|
|
||||||
if (rows.length && rows[0].count != 0) {
|
|
||||||
log(LM_WARN, ` Destination table already exists (found ${rows[0].count} items), nothing to do`);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const dbSrc = await storagePool.get();
|
|
||||||
try {
|
|
||||||
const rows = await dbSrc.all(`SELECT * FROM storage`);
|
|
||||||
await db.insert({table: 'storage', rows});
|
|
||||||
log(` Inserted ${rows.length} items`);
|
|
||||||
} finally {
|
|
||||||
dbSrc.ret();
|
|
||||||
}
|
|
||||||
} finally {
|
|
||||||
log('Converter finish');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
module.exports = Converter;
|
|
||||||
@@ -31,7 +31,29 @@ class JembaConnManager {
|
|||||||
|
|
||||||
ayncExit.add(this.close.bind(this));
|
ayncExit.add(this.close.bind(this));
|
||||||
|
|
||||||
|
const serverModes = new Set();
|
||||||
|
for (const serverCfg of this.config.servers) {
|
||||||
|
serverModes.add(serverCfg.mode);
|
||||||
|
}
|
||||||
|
|
||||||
for (const dbConfig of this.config.jembaDb) {
|
for (const dbConfig of this.config.jembaDb) {
|
||||||
|
//проверка, надо ли открывать базу, зависит от serverMode
|
||||||
|
if (dbConfig.serverMode) {
|
||||||
|
let serverMode = dbConfig.serverMode;
|
||||||
|
if (!Array.isArray(dbConfig.serverMode))
|
||||||
|
serverMode = [dbConfig.serverMode];
|
||||||
|
|
||||||
|
let modePresent = false;
|
||||||
|
for (const mode of serverMode) {
|
||||||
|
modePresent = serverModes.has(mode);
|
||||||
|
if (modePresent)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!modePresent)
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
const dbPath = `${this.config.dataDir}/db/${dbConfig.dbName}`;
|
const dbPath = `${this.config.dataDir}/db/${dbConfig.dbName}`;
|
||||||
|
|
||||||
//бэкап
|
//бэкап
|
||||||
|
|||||||
@@ -1,193 +0,0 @@
|
|||||||
//TODO: удалить модуль в 2023г
|
|
||||||
const sqlite3 = require('sqlite3');
|
|
||||||
const sqlite = require('sqlite');
|
|
||||||
|
|
||||||
const SQL = require('sql-template-strings');
|
|
||||||
|
|
||||||
class SqliteConnectionPool {
|
|
||||||
constructor() {
|
|
||||||
this.closed = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
async open(poolConfig, dbFileName) {
|
|
||||||
const connCount = poolConfig.connCount || 1;
|
|
||||||
const busyTimeout = poolConfig.busyTimeout || 60*1000;
|
|
||||||
const cacheSize = poolConfig.cacheSize || 2000;
|
|
||||||
|
|
||||||
this.dbFileName = dbFileName;
|
|
||||||
this.connections = [];
|
|
||||||
this.freed = new Set();
|
|
||||||
this.waitingQueue = [];
|
|
||||||
|
|
||||||
for (let i = 0; i < connCount; i++) {
|
|
||||||
let client = await sqlite.open({
|
|
||||||
filename: dbFileName,
|
|
||||||
driver: sqlite3.Database
|
|
||||||
});
|
|
||||||
|
|
||||||
client.configure('busyTimeout', busyTimeout); //ms
|
|
||||||
await client.exec(`PRAGMA cache_size = ${cacheSize}`);
|
|
||||||
|
|
||||||
client.ret = () => {
|
|
||||||
this.freed.add(i);
|
|
||||||
if (this.waitingQueue.length) {
|
|
||||||
this.waitingQueue.shift().onFreed(i);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
this.freed.add(i);
|
|
||||||
this.connections[i] = client;
|
|
||||||
}
|
|
||||||
this.closed = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
get() {
|
|
||||||
return new Promise((resolve) => {
|
|
||||||
if (this.closed)
|
|
||||||
throw new Error('Connection pool closed');
|
|
||||||
|
|
||||||
const freeConnIndex = this.freed.values().next().value;
|
|
||||||
if (freeConnIndex !== undefined) {
|
|
||||||
this.freed.delete(freeConnIndex);
|
|
||||||
resolve(this.connections[freeConnIndex]);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
this.waitingQueue.push({
|
|
||||||
onFreed: (connIndex) => {
|
|
||||||
this.freed.delete(connIndex);
|
|
||||||
resolve(this.connections[connIndex]);
|
|
||||||
},
|
|
||||||
});
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
async run(query) {
|
|
||||||
const dbh = await this.get();
|
|
||||||
try {
|
|
||||||
let result = await dbh.run(query);
|
|
||||||
dbh.ret();
|
|
||||||
return result;
|
|
||||||
} catch (e) {
|
|
||||||
dbh.ret();
|
|
||||||
throw e;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async all(query) {
|
|
||||||
const dbh = await this.get();
|
|
||||||
try {
|
|
||||||
let result = await dbh.all(query);
|
|
||||||
dbh.ret();
|
|
||||||
return result;
|
|
||||||
} catch (e) {
|
|
||||||
dbh.ret();
|
|
||||||
throw e;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async exec(query) {
|
|
||||||
const dbh = await this.get();
|
|
||||||
try {
|
|
||||||
let result = await dbh.exec(query);
|
|
||||||
dbh.ret();
|
|
||||||
return result;
|
|
||||||
} catch (e) {
|
|
||||||
dbh.ret();
|
|
||||||
throw e;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async close() {
|
|
||||||
for (let i = 0; i < this.connections.length; i++) {
|
|
||||||
await this.connections[i].close();
|
|
||||||
}
|
|
||||||
this.closed = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Modified from node-sqlite/.../src/Database.js
|
|
||||||
async migrate(migs, table, force) {
|
|
||||||
const migrations = migs.sort((a, b) => Math.sign(a.id - b.id));
|
|
||||||
|
|
||||||
if (!migrations.length) {
|
|
||||||
throw new Error('No migration data');
|
|
||||||
}
|
|
||||||
|
|
||||||
migrations.map(migration => {
|
|
||||||
const data = migration.data;
|
|
||||||
const [up, down] = data.split(/^--\s+?down\b/mi);
|
|
||||||
if (!down) {
|
|
||||||
const message = `The ${migration.filename} file does not contain '-- Down' separator.`;
|
|
||||||
throw new Error(message);
|
|
||||||
} else {
|
|
||||||
/* eslint-disable no-param-reassign */
|
|
||||||
migration.up = up.replace(/^-- .*?$/gm, '').trim();// Remove comments
|
|
||||||
migration.down = down.trim(); // and trim whitespaces
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
// Create a database table for migrations meta data if it doesn't exist
|
|
||||||
await this.run(`CREATE TABLE IF NOT EXISTS "${table}" (
|
|
||||||
id INTEGER PRIMARY KEY,
|
|
||||||
name TEXT NOT NULL,
|
|
||||||
up TEXT NOT NULL,
|
|
||||||
down TEXT NOT NULL
|
|
||||||
)`);
|
|
||||||
|
|
||||||
// Get the list of already applied migrations
|
|
||||||
let dbMigrations = await this.all(
|
|
||||||
`SELECT id, name, up, down FROM "${table}" ORDER BY id ASC`,
|
|
||||||
);
|
|
||||||
|
|
||||||
// Undo migrations that exist only in the database but not in migs,
|
|
||||||
// also undo the last migration if the `force` option was set to `last`.
|
|
||||||
const lastMigration = migrations[migrations.length - 1];
|
|
||||||
for (const migration of dbMigrations.slice().sort((a, b) => Math.sign(b.id - a.id))) {
|
|
||||||
if (!migrations.some(x => x.id === migration.id) ||
|
|
||||||
(force === 'last' && migration.id === lastMigration.id)) {
|
|
||||||
const dbh = await this.get();
|
|
||||||
await dbh.run('BEGIN');
|
|
||||||
try {
|
|
||||||
await dbh.exec(migration.down);
|
|
||||||
await dbh.run(SQL`DELETE FROM "`.append(table).append(SQL`" WHERE id = ${migration.id}`));
|
|
||||||
await dbh.run('COMMIT');
|
|
||||||
dbMigrations = dbMigrations.filter(x => x.id !== migration.id);
|
|
||||||
} catch (err) {
|
|
||||||
await dbh.run('ROLLBACK');
|
|
||||||
throw err;
|
|
||||||
} finally {
|
|
||||||
dbh.ret();
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Apply pending migrations
|
|
||||||
let applied = [];
|
|
||||||
const lastMigrationId = dbMigrations.length ? dbMigrations[dbMigrations.length - 1].id : 0;
|
|
||||||
for (const migration of migrations) {
|
|
||||||
if (migration.id > lastMigrationId) {
|
|
||||||
const dbh = await this.get();
|
|
||||||
await dbh.run('BEGIN');
|
|
||||||
try {
|
|
||||||
await dbh.exec(migration.up);
|
|
||||||
await dbh.run(SQL`INSERT INTO "`.append(table).append(
|
|
||||||
SQL`" (id, name, up, down) VALUES (${migration.id}, ${migration.name}, ${migration.up}, ${migration.down})`)
|
|
||||||
);
|
|
||||||
await dbh.run('COMMIT');
|
|
||||||
applied.push(migration.id);
|
|
||||||
} catch (err) {
|
|
||||||
await dbh.run('ROLLBACK');
|
|
||||||
throw err;
|
|
||||||
} finally {
|
|
||||||
dbh.ret();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return applied;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
module.exports = SqliteConnectionPool;
|
|
||||||
12
server/db/jembaMigrations/app/001-create.js
Normal file
12
server/db/jembaMigrations/app/001-create.js
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
module.exports = {
|
||||||
|
up: [
|
||||||
|
['create', {
|
||||||
|
table: 'remote_sent'
|
||||||
|
}],
|
||||||
|
],
|
||||||
|
down: [
|
||||||
|
['drop', {
|
||||||
|
table: 'remote_sent'
|
||||||
|
}],
|
||||||
|
]
|
||||||
|
};
|
||||||
22
server/db/jembaMigrations/app/002-create.js
Normal file
22
server/db/jembaMigrations/app/002-create.js
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
module.exports = {
|
||||||
|
up: [
|
||||||
|
['create', {
|
||||||
|
/*{
|
||||||
|
id, // book URL
|
||||||
|
queryTime: Number,
|
||||||
|
checkTime: Number, // 0 - never checked
|
||||||
|
modTime: String,
|
||||||
|
size: Number,
|
||||||
|
checkSum: String, //sha256
|
||||||
|
state: Number, // 0 - not processing, 1 - processing
|
||||||
|
error: String,
|
||||||
|
}*/
|
||||||
|
table: 'buc'
|
||||||
|
}],
|
||||||
|
],
|
||||||
|
down: [
|
||||||
|
['drop', {
|
||||||
|
table: 'buc'
|
||||||
|
}],
|
||||||
|
]
|
||||||
|
};
|
||||||
7
server/db/jembaMigrations/app/index.js
Normal file
7
server/db/jembaMigrations/app/index.js
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
module.exports = {
|
||||||
|
table: 'migration1',
|
||||||
|
data: [
|
||||||
|
{id: 1, name: 'create', data: require('./001-create')},
|
||||||
|
{id: 2, name: 'create', data: require('./002-create')},
|
||||||
|
]
|
||||||
|
}
|
||||||
29
server/db/jembaMigrations/book-update-server/001-create.js
Normal file
29
server/db/jembaMigrations/book-update-server/001-create.js
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
module.exports = {
|
||||||
|
up: [
|
||||||
|
['create', {
|
||||||
|
/*{
|
||||||
|
id, // book URL
|
||||||
|
queryTime: Number,
|
||||||
|
checkTime: Number, // 0 - never checked
|
||||||
|
modTime: String,
|
||||||
|
size: Number,
|
||||||
|
checkSum: String, //sha256
|
||||||
|
state: Number, // 0 - not processing, 1 - processing
|
||||||
|
error: String,
|
||||||
|
}*/
|
||||||
|
table: 'buc',
|
||||||
|
flag: [
|
||||||
|
{name: 'notProcessing', check: `(r) => r.state === 0`},
|
||||||
|
],
|
||||||
|
index: [
|
||||||
|
{field: 'queryTime', type: 'number'},
|
||||||
|
{field: 'checkTime', type: 'number'},
|
||||||
|
]
|
||||||
|
}],
|
||||||
|
],
|
||||||
|
down: [
|
||||||
|
['drop', {
|
||||||
|
table: 'buc'
|
||||||
|
}],
|
||||||
|
]
|
||||||
|
};
|
||||||
@@ -1,4 +1,5 @@
|
|||||||
module.exports = {
|
module.exports = {
|
||||||
//'app': require('./jembaMigrations/app'),
|
'app': require('./app'),
|
||||||
'reader-storage': require('./reader-storage'),
|
'reader-storage': require('./reader-storage'),
|
||||||
|
'book-update-server': require('./book-update-server'),
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,5 +0,0 @@
|
|||||||
module.exports = {
|
|
||||||
table: 'migration1',
|
|
||||||
data: [
|
|
||||||
]
|
|
||||||
}
|
|
||||||
@@ -1,7 +0,0 @@
|
|||||||
module.exports = `
|
|
||||||
-- Up
|
|
||||||
CREATE TABLE storage (id TEXT PRIMARY KEY, rev INTEGER, time INTEGER, data TEXT);
|
|
||||||
|
|
||||||
-- Down
|
|
||||||
DROP TABLE storage;
|
|
||||||
`;
|
|
||||||
@@ -1,16 +1,19 @@
|
|||||||
require('tls').DEFAULT_MIN_VERSION = 'TLSv1';
|
require('tls').DEFAULT_MIN_VERSION = 'TLSv1';
|
||||||
|
|
||||||
const fs = require('fs-extra');
|
const fs = require('fs-extra');
|
||||||
const path = require('path');
|
|
||||||
const argv = require('minimist')(process.argv.slice(2));
|
const argv = require('minimist')(process.argv.slice(2));
|
||||||
const express = require('express');
|
const express = require('express');
|
||||||
const compression = require('compression');
|
const compression = require('compression');
|
||||||
const http = require('http');
|
const http = require('http');
|
||||||
|
const https = require('https');
|
||||||
const WebSocket = require ('ws');
|
const WebSocket = require ('ws');
|
||||||
|
|
||||||
const ayncExit = new (require('./core/AsyncExit'))();
|
const ayncExit = new (require('./core/AsyncExit'))();
|
||||||
|
|
||||||
let log = null;
|
let log = null;
|
||||||
|
|
||||||
|
const maxPayloadSize = 50;//in MB
|
||||||
|
|
||||||
async function init() {
|
async function init() {
|
||||||
//config
|
//config
|
||||||
const configManager = new (require('./config'))();//singleton
|
const configManager = new (require('./config'))();//singleton
|
||||||
@@ -43,15 +46,8 @@ async function init() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
//connections
|
//connections
|
||||||
const connManager = new (require('./db/ConnManager'))();//singleton
|
|
||||||
await connManager.init(config);
|
|
||||||
|
|
||||||
const jembaConnManager = new (require('./db/JembaConnManager'))();//singleton
|
const jembaConnManager = new (require('./db/JembaConnManager'))();//singleton
|
||||||
await jembaConnManager.init(config, argv['auto-repair']);
|
await jembaConnManager.init(config, argv['auto-repair']);
|
||||||
|
|
||||||
//converter SQLITE => JembaDb
|
|
||||||
const converter = new (require('./db/Converter'))();
|
|
||||||
await converter.run(config);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
async function main() {
|
async function main() {
|
||||||
@@ -62,8 +58,16 @@ async function main() {
|
|||||||
for (let serverCfg of config.servers) {
|
for (let serverCfg of config.servers) {
|
||||||
if (serverCfg.mode !== 'none') {
|
if (serverCfg.mode !== 'none') {
|
||||||
const app = express();
|
const app = express();
|
||||||
const server = http.createServer(app);
|
let server;
|
||||||
const wss = new WebSocket.Server({ server, maxPayload: 10*1024*1024 });
|
if (serverCfg.isHttps) {
|
||||||
|
const key = fs.readFileSync(`${config.dataDir}/${serverCfg.keysFile}.key`);
|
||||||
|
const cert = fs.readFileSync(`${config.dataDir}/${serverCfg.keysFile}.crt`);
|
||||||
|
|
||||||
|
server = https.createServer({key, cert}, app);
|
||||||
|
} else {
|
||||||
|
server = http.createServer(app);
|
||||||
|
}
|
||||||
|
const wss = new WebSocket.Server({ server, maxPayload: maxPayloadSize*1024*1024 });
|
||||||
|
|
||||||
const serverConfig = Object.assign({}, config, serverCfg);
|
const serverConfig = Object.assign({}, config, serverCfg);
|
||||||
|
|
||||||
@@ -75,20 +79,10 @@ async function main() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
app.use(compression({ level: 1 }));
|
app.use(compression({ level: 1 }));
|
||||||
app.use(express.json({limit: '10mb'}));
|
app.use(express.json({limit: `${maxPayloadSize}mb`}));
|
||||||
if (devModule)
|
if (devModule)
|
||||||
devModule.logQueries(app);
|
devModule.logQueries(app);
|
||||||
|
|
||||||
app.use(express.static(serverConfig.publicDir, {
|
|
||||||
maxAge: '30d',
|
|
||||||
setHeaders: (res, filePath) => {
|
|
||||||
if (path.basename(path.dirname(filePath)) == 'tmp') {
|
|
||||||
res.set('Content-Type', 'application/xml');
|
|
||||||
res.set('Content-Encoding', 'gzip');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}));
|
|
||||||
|
|
||||||
require('./routes').initRoutes(app, wss, serverConfig);
|
require('./routes').initRoutes(app, wss, serverConfig);
|
||||||
|
|
||||||
if (devModule) {
|
if (devModule) {
|
||||||
@@ -101,7 +95,7 @@ async function main() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
server.listen(serverConfig.port, serverConfig.ip, function() {
|
server.listen(serverConfig.port, serverConfig.ip, function() {
|
||||||
log(`Server-${serverConfig.serverName} is ready on ${serverConfig.ip}:${serverConfig.port}, mode: ${serverConfig.mode}`);
|
log(`Server "${serverConfig.serverName}" is ready on ${(serverConfig.isHttps ? 'https://' : 'http://')}${serverConfig.ip}:${serverConfig.port}, mode: ${serverConfig.mode}`);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,8 +1,24 @@
|
|||||||
const c = require('./controllers');
|
const fs = require('fs-extra');
|
||||||
const utils = require('./core/utils');
|
const path = require('path');
|
||||||
|
|
||||||
|
const express = require('express');
|
||||||
const multer = require('multer');
|
const multer = require('multer');
|
||||||
|
|
||||||
|
const ReaderWorker = require('./core/Reader/ReaderWorker');//singleton
|
||||||
|
const log = new (require('./core/AppLogger'))().log;//singleton
|
||||||
|
|
||||||
|
const c = require('./controllers');
|
||||||
|
const utils = require('./core/utils');
|
||||||
|
|
||||||
function initRoutes(app, wss, config) {
|
function initRoutes(app, wss, config) {
|
||||||
|
//эксклюзив для update_checker
|
||||||
|
if (config.mode === 'book_update_checker') {
|
||||||
|
new c.BookUpdateCheckerController(wss, config);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
initStatic(app, config);
|
||||||
|
|
||||||
const misc = new c.MiscController(config);
|
const misc = new c.MiscController(config);
|
||||||
const reader = new c.ReaderController(config);
|
const reader = new c.ReaderController(config);
|
||||||
const worker = new c.WorkerController(config);
|
const worker = new c.WorkerController(config);
|
||||||
@@ -29,7 +45,6 @@ function initRoutes(app, wss, config) {
|
|||||||
['POST', '/api/reader/load-book', reader.loadBook.bind(reader), [aAll], {}],
|
['POST', '/api/reader/load-book', reader.loadBook.bind(reader), [aAll], {}],
|
||||||
['POST', '/api/reader/storage', reader.storage.bind(reader), [aAll], {}],
|
['POST', '/api/reader/storage', reader.storage.bind(reader), [aAll], {}],
|
||||||
['POST', '/api/reader/upload-file', [upload.single('file'), reader.uploadFile.bind(reader)], [aAll], {}],
|
['POST', '/api/reader/upload-file', [upload.single('file'), reader.uploadFile.bind(reader)], [aAll], {}],
|
||||||
['POST', '/api/reader/restore-cached-file', reader.restoreCachedFile.bind(reader), [aAll], {}],
|
|
||||||
['POST', '/api/worker/get-state', worker.getState.bind(worker), [aAll], {}],
|
['POST', '/api/worker/get-state', worker.getState.bind(worker), [aAll], {}],
|
||||||
];
|
];
|
||||||
|
|
||||||
@@ -77,6 +92,48 @@ function initRoutes(app, wss, config) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function initStatic(app, config) {
|
||||||
|
const readerWorker = new ReaderWorker(config);
|
||||||
|
|
||||||
|
//восстановление файлов в /tmp и /upload из webdav-storage, при необходимости
|
||||||
|
app.use(async(req, res, next) => {
|
||||||
|
if ((req.method !== 'GET' && req.method !== 'HEAD') ||
|
||||||
|
!(req.path.indexOf('/tmp/') === 0 || req.path.indexOf('/upload/') === 0)
|
||||||
|
) {
|
||||||
|
return next();
|
||||||
|
}
|
||||||
|
|
||||||
|
const filePath = `${config.publicDir}${req.path}`;
|
||||||
|
|
||||||
|
//восстановим
|
||||||
|
try {
|
||||||
|
if (!await fs.pathExists(filePath)) {
|
||||||
|
if (req.path.indexOf('/tmp/') === 0) {
|
||||||
|
await readerWorker.restoreRemoteFile(req.path, '/tmp');
|
||||||
|
} else if (req.path.indexOf('/upload/') === 0) {
|
||||||
|
await readerWorker.restoreRemoteFile(req.path, '/upload');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch(e) {
|
||||||
|
log(LM_ERR, `static::restoreRemoteFile ${req.path} > ${e.message}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
return next();
|
||||||
|
});
|
||||||
|
|
||||||
|
const tmpDir = `${config.publicDir}/tmp`;
|
||||||
|
app.use(express.static(config.publicDir, {
|
||||||
|
maxAge: '30d',
|
||||||
|
|
||||||
|
setHeaders: (res, filePath) => {
|
||||||
|
if (path.dirname(filePath) == tmpDir) {
|
||||||
|
res.set('Content-Type', 'application/xml');
|
||||||
|
res.set('Content-Encoding', 'gzip');
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
initRoutes
|
initRoutes
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user