Compare commits
173 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
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 | ||
|
|
873a08fee1 | ||
|
|
7e89228803 | ||
|
|
fc630923a4 | ||
|
|
928f911d03 | ||
|
|
7ffcd3fe1b | ||
|
|
0efbaf643a | ||
|
|
f1bf8e54ae | ||
|
|
b4aa6ab6c8 | ||
|
|
72431f0202 | ||
|
|
04a326c0e4 | ||
|
|
931966f4f3 | ||
|
|
8808cc4779 | ||
|
|
988c959eba | ||
|
|
c0b658d9e6 | ||
|
|
3190246f34 | ||
|
|
d957b4a5f9 | ||
|
|
bef9e5705c | ||
|
|
eb2affa518 | ||
|
|
07b9a3c033 | ||
|
|
3ca14ae06a | ||
|
|
7caa0c2112 | ||
|
|
9c69f5bc01 | ||
|
|
125a2e0f17 | ||
|
|
1b4360b897 | ||
|
|
4775d6e47b | ||
|
|
33fc553c55 | ||
|
|
25cad81c50 | ||
|
|
02a2099c1f | ||
|
|
1cda186b1a | ||
|
|
f10291b6c6 | ||
|
|
26ab5d6765 | ||
|
|
5edeed0747 | ||
|
|
c878ce432f | ||
|
|
81798897c8 | ||
|
|
63840fadbc | ||
|
|
36aa057035 | ||
|
|
30afd2421c | ||
|
|
53a1d90bd8 | ||
|
|
2ecf6beef2 | ||
|
|
85910a20e9 | ||
|
|
66cf7790b3 | ||
|
|
4a9eb7e4bb | ||
|
|
07446696c1 | ||
|
|
a29f9d9a4b | ||
|
|
d49c9baec3 | ||
|
|
8c9d4a12ee | ||
|
|
fce69e4657 | ||
|
|
b387509f88 | ||
|
|
8dc8bdc0d6 | ||
|
|
00caae8363 | ||
|
|
2ead8570a7 | ||
|
|
408315466b | ||
|
|
c651836554 | ||
|
|
03a1e70fce | ||
|
|
ab5a11a24f | ||
|
|
8cd6ed472c | ||
|
|
055181b744 | ||
|
|
e331a3920b | ||
|
|
c62bccb470 | ||
|
|
ea351ea293 | ||
|
|
d806a07c60 | ||
|
|
c0ea096f1f | ||
|
|
011d4a1672 | ||
|
|
4836a737c6 | ||
|
|
5712b2ee17 | ||
|
|
32dd17694e | ||
|
|
3ebc932a6a | ||
|
|
8f351d9bef | ||
|
|
5ae3ea94e4 | ||
|
|
f203d453a4 | ||
|
|
0d5cba121b | ||
|
|
0cd6a48a46 | ||
|
|
4e07ce2b5c | ||
|
|
85a525e301 | ||
|
|
03e4a6d723 | ||
|
|
ab28af1abe | ||
|
|
7fceed5301 | ||
|
|
0077816afa | ||
|
|
cb01423147 | ||
|
|
61b0712d36 | ||
|
|
12d7843377 | ||
|
|
9293c0a0d4 | ||
|
|
bb9522197a | ||
|
|
450a2e0664 | ||
|
|
41e35f3ec8 | ||
|
|
a9bc98abe3 | ||
|
|
47bca03532 | ||
|
|
942021371c | ||
|
|
ea2f178730 | ||
|
|
4b5c8d9efe | ||
|
|
28ebf13c3a | ||
|
|
5d52e63dd9 | ||
|
|
1a0e024050 | ||
|
|
e627a0d970 | ||
|
|
48668d94ad | ||
|
|
e08c431dd9 | ||
|
|
5ee58ad6f0 | ||
|
|
ac0a4f0586 | ||
|
|
b6f4c153e5 | ||
|
|
4fdaf5f555 | ||
|
|
b4ee9d6c00 | ||
|
|
7c73c74730 | ||
|
|
c20aa089fa | ||
|
|
b0e15c22ea | ||
|
|
d58a2c065a | ||
|
|
53135e7ee8 | ||
|
|
5c48ca9e6c | ||
|
|
c4a280f3d8 | ||
|
|
ba2943c722 | ||
|
|
26f6ffc83a | ||
|
|
bcf075a72c | ||
|
|
02d458d192 | ||
|
|
a349d8af68 | ||
|
|
0dbaf32aac | ||
|
|
e8c41ef3a8 | ||
|
|
e43a44e986 | ||
|
|
f14b8ed277 | ||
|
|
bbfe8a64cb | ||
|
|
bcf3c2dab0 | ||
|
|
d5404fd260 | ||
|
|
54bc662e43 | ||
|
|
42546ca97e | ||
|
|
5c13cf0eb9 | ||
|
|
2a9d44ae9a | ||
|
|
38414ae7b6 | ||
|
|
3ecb3e80ac | ||
|
|
4968828488 | ||
|
|
4db3cd24df | ||
|
|
45c6d3da77 | ||
|
|
4aab1da3c6 | ||
|
|
bf5dfa1c15 | ||
|
|
7549bdd2b4 | ||
|
|
1bb2525ab2 |
@@ -12,6 +12,7 @@
|
|||||||
"@babel"
|
"@babel"
|
||||||
],
|
],
|
||||||
"env": {
|
"env": {
|
||||||
|
"es6": true,
|
||||||
"browser": true,
|
"browser": true,
|
||||||
"node": true
|
"node": true
|
||||||
},
|
},
|
||||||
@@ -30,6 +31,7 @@
|
|||||||
"vue/max-attributes-per-line": "off",
|
"vue/max-attributes-per-line": "off",
|
||||||
"vue/html-self-closing": "off",
|
"vue/html-self-closing": "off",
|
||||||
"vue/no-v-html": "off",
|
"vue/no-v-html": "off",
|
||||||
|
"vue/no-v-model-argument": "off",
|
||||||
|
|
||||||
"strict": 0,
|
"strict": 0,
|
||||||
"indent": [0, 4, {
|
"indent": [0, 4, {
|
||||||
|
|||||||
@@ -4,7 +4,7 @@ const util = require('util');
|
|||||||
const stream = require('stream');
|
const stream = require('stream');
|
||||||
const pipeline = util.promisify(stream.pipeline);
|
const pipeline = util.promisify(stream.pipeline);
|
||||||
|
|
||||||
const got = require('got');
|
const axios = require('axios');
|
||||||
const FileDecompressor = require('../server/core/FileDecompressor');
|
const FileDecompressor = require('../server/core/FileDecompressor');
|
||||||
|
|
||||||
const distDir = path.resolve(__dirname, '../dist');
|
const distDir = path.resolve(__dirname, '../dist');
|
||||||
@@ -29,7 +29,8 @@ async function main() {
|
|||||||
|
|
||||||
if (!await fs.pathExists(sqliteDecompressedFilename)) {
|
if (!await fs.pathExists(sqliteDecompressedFilename)) {
|
||||||
// Скачиваем node_sqlite3.node для винды, т.к. pkg не включает его в сборку
|
// Скачиваем node_sqlite3.node для винды, т.к. pkg не включает его в сборку
|
||||||
await pipeline(got.stream(sqliteRemoteUrl), fs.createWriteStream(`${tempDownloadDir}/sqlite.tar.gz`));
|
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(`done downloading ${sqliteRemoteUrl}`);
|
||||||
|
|
||||||
//распаковываем
|
//распаковываем
|
||||||
@@ -46,7 +47,8 @@ async function main() {
|
|||||||
// Скачиваем ipfs
|
// Скачиваем ipfs
|
||||||
const ipfsRemoteUrl = 'https://dist.ipfs.io/go-ipfs/v0.4.18/go-ipfs_v0.4.18_linux-amd64.tar.gz';
|
const ipfsRemoteUrl = 'https://dist.ipfs.io/go-ipfs/v0.4.18/go-ipfs_v0.4.18_linux-amd64.tar.gz';
|
||||||
|
|
||||||
await pipeline(got.stream(ipfsRemoteUrl), fs.createWriteStream(`${tempDownloadDir}/ipfs.tar.gz`));
|
const res = await axios.get(ipfsRemoteUrl, {responseType: 'stream'})
|
||||||
|
await pipeline(res.data, fs.createWriteStream(`${tempDownloadDir}/ipfs.tar.gz`));
|
||||||
console.log(`done downloading ${ipfsRemoteUrl}`);
|
console.log(`done downloading ${ipfsRemoteUrl}`);
|
||||||
|
|
||||||
//распаковываем
|
//распаковываем
|
||||||
|
|||||||
@@ -5,11 +5,12 @@ const { VueLoaderPlugin } = require('vue-loader');
|
|||||||
const clientDir = path.resolve(__dirname, '../client');
|
const clientDir = path.resolve(__dirname, '../client');
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
/*resolve: {
|
resolve: {
|
||||||
alias: {
|
alias: {
|
||||||
vue: '@vue/compat'
|
ws: false,
|
||||||
|
//vue: '@vue/compat'
|
||||||
}
|
}
|
||||||
},*/
|
},
|
||||||
entry: [`${clientDir}/main.js`],
|
entry: [`${clientDir}/main.js`],
|
||||||
output: {
|
output: {
|
||||||
publicPath: '/app/',
|
publicPath: '/app/',
|
||||||
@@ -62,34 +63,6 @@ module.exports = {
|
|||||||
filename: 'fonts/[name]-[hash:6][ext]'
|
filename: 'fonts/[name]-[hash:6][ext]'
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
/*{
|
|
||||||
test: /\.gif$/,
|
|
||||||
loader: "url-loader",
|
|
||||||
options: {
|
|
||||||
name: "images/[name]-[hash:6].[ext]"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
test: /\.png$/,
|
|
||||||
loader: "url-loader",
|
|
||||||
options: {
|
|
||||||
name: "images/[name]-[hash:6].[ext]"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
test: /\.jpg$/,
|
|
||||||
loader: "file-loader",
|
|
||||||
options: {
|
|
||||||
name: "images/[name]-[hash:6].[ext]"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
test: /\.(ttf|eot|woff|woff2)$/,
|
|
||||||
loader: "file-loader",
|
|
||||||
options: {
|
|
||||||
name: "fonts/[name]-[hash:6].[ext]"
|
|
||||||
}
|
|
||||||
},*/
|
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
|
|
||||||
|
|||||||
@@ -4,7 +4,7 @@ const util = require('util');
|
|||||||
const stream = require('stream');
|
const stream = require('stream');
|
||||||
const pipeline = util.promisify(stream.pipeline);
|
const pipeline = util.promisify(stream.pipeline);
|
||||||
|
|
||||||
const got = require('got');
|
const axios = require('axios');
|
||||||
const FileDecompressor = require('../server/core/FileDecompressor');
|
const FileDecompressor = require('../server/core/FileDecompressor');
|
||||||
|
|
||||||
const distDir = path.resolve(__dirname, '../dist');
|
const distDir = path.resolve(__dirname, '../dist');
|
||||||
@@ -29,7 +29,8 @@ async function main() {
|
|||||||
|
|
||||||
if (!await fs.pathExists(sqliteDecompressedFilename)) {
|
if (!await fs.pathExists(sqliteDecompressedFilename)) {
|
||||||
// Скачиваем node_sqlite3.node для винды, т.к. pkg не включает его в сборку
|
// Скачиваем node_sqlite3.node для винды, т.к. pkg не включает его в сборку
|
||||||
await pipeline(got.stream(sqliteRemoteUrl), fs.createWriteStream(`${tempDownloadDir}/sqlite.tar.gz`));
|
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(`done downloading ${sqliteRemoteUrl}`);
|
||||||
|
|
||||||
//распаковываем
|
//распаковываем
|
||||||
@@ -46,7 +47,8 @@ async function main() {
|
|||||||
// Скачиваем ipfs
|
// Скачиваем ipfs
|
||||||
const ipfsRemoteUrl = 'https://dist.ipfs.io/go-ipfs/v0.4.18/go-ipfs_v0.4.18_windows-amd64.zip';
|
const ipfsRemoteUrl = 'https://dist.ipfs.io/go-ipfs/v0.4.18/go-ipfs_v0.4.18_windows-amd64.zip';
|
||||||
|
|
||||||
await pipeline(got.stream(ipfsRemoteUrl), fs.createWriteStream(`${tempDownloadDir}/ipfs.zip`));
|
const res = await axios.get(ipfsRemoteUrl, {responseType: 'stream'})
|
||||||
|
await pipeline(res.data, fs.createWriteStream(`${tempDownloadDir}/ipfs.zip`));
|
||||||
console.log(`done downloading ${ipfsRemoteUrl}`);
|
console.log(`done downloading ${ipfsRemoteUrl}`);
|
||||||
|
|
||||||
//распаковываем
|
//распаковываем
|
||||||
|
|||||||
@@ -9,7 +9,7 @@ class Misc {
|
|||||||
async loadConfig() {
|
async loadConfig() {
|
||||||
|
|
||||||
const query = {params: [
|
const query = {params: [
|
||||||
'name', 'version', 'mode', 'maxUploadFileSize', 'useExternalBookConverter', 'branch',
|
'name', 'version', 'mode', 'maxUploadFileSize', 'useExternalBookConverter', 'acceptFileExt', '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({
|
||||||
@@ -174,11 +175,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);
|
||||||
|
|
||||||
@@ -219,12 +219,39 @@ class Reader {
|
|||||||
const state = response.state;
|
const state = response.state;
|
||||||
if (!state)
|
if (!state)
|
||||||
throw new Error('Неверный ответ api');
|
throw new Error('Неверный ответ api');
|
||||||
if (response.state == 'error') {
|
if (state == 'error') {
|
||||||
throw new Error(response.error);
|
throw new Error(response.error);
|
||||||
}
|
}
|
||||||
|
|
||||||
return response;
|
return response;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async uploadFileBuf(buf, urlCallback) {
|
||||||
|
const key = utils.toHex(cryptoUtils.sha256(buf));
|
||||||
|
const url = `disk://${key}`;
|
||||||
|
|
||||||
|
if (urlCallback)
|
||||||
|
urlCallback(url);
|
||||||
|
|
||||||
|
let response;
|
||||||
|
try {
|
||||||
|
await axios.head(`/upload/${key}`, {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;
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export default new Reader();
|
export default new Reader();
|
||||||
@@ -11,7 +11,7 @@
|
|||||||
Открыть выбранную закладку
|
Открыть выбранную закладку
|
||||||
</q-tooltip>
|
</q-tooltip>
|
||||||
</q-btn>
|
</q-btn>
|
||||||
<q-input ref="search" v-model="search" class="col" rounded outlined dense bg-color="white" placeholder="Найти">
|
<q-input ref="search" v-model="search" class="col" outlined dense bg-color="white" placeholder="Найти">
|
||||||
<template #append>
|
<template #append>
|
||||||
<q-icon v-if="search !== ''" name="la la-times" class="cursor-pointer" @click="resetSearch" />
|
<q-icon v-if="search !== ''" name="la la-times" class="cursor-pointer" @click="resetSearch" />
|
||||||
</template>
|
</template>
|
||||||
@@ -55,16 +55,16 @@
|
|||||||
|
|
||||||
<div class="col fit tree">
|
<div class="col fit tree">
|
||||||
<div v-show="nodes.length" class="checkbox-tick-all">
|
<div v-show="nodes.length" class="checkbox-tick-all">
|
||||||
<q-checkbox v-model="tickAll" size="36px" label="Выбрать все" @input="makeTickAll" />
|
<q-checkbox v-model="tickAll" size="36px" label="Выбрать все" @update:model-value="makeTickAll" />
|
||||||
</div>
|
</div>
|
||||||
<q-tree
|
<q-tree
|
||||||
|
v-model:selected="selected"
|
||||||
|
v-model:ticked="ticked"
|
||||||
|
v-model:expanded="expanded"
|
||||||
class="q-my-xs"
|
class="q-my-xs"
|
||||||
:nodes="nodes"
|
:nodes="nodes"
|
||||||
node-key="key"
|
node-key="key"
|
||||||
tick-strategy="leaf"
|
tick-strategy="leaf"
|
||||||
v-model:selected="selected"
|
|
||||||
v-model:ticked="ticked"
|
|
||||||
v-model:expanded="expanded"
|
|
||||||
selected-color="black"
|
selected-color="black"
|
||||||
:filter="search"
|
:filter="search"
|
||||||
no-nodes-label="Закладок пока нет"
|
no-nodes-label="Закладок пока нет"
|
||||||
@@ -97,7 +97,7 @@ const componentOptions = {
|
|||||||
Window,
|
Window,
|
||||||
},
|
},
|
||||||
watch: {
|
watch: {
|
||||||
ticked: function() {
|
ticked() {
|
||||||
this.checkAllTicked();
|
this.checkAllTicked();
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,19 +5,19 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<template #buttons>
|
<template #buttons>
|
||||||
<span class="full-screen-button row justify-center items-center" @mousedown.stop @click="fullScreenToggle">
|
<span class="header-button row justify-center items-center" @mousedown.stop @click="fullScreenToggle">
|
||||||
<q-icon :name="(fullScreenActive ? 'la la-compress-arrows-alt': 'la la-expand-arrows-alt')" size="16px" />
|
<q-icon :name="(fullScreenActive ? 'la la-compress-arrows-alt': 'la la-expand-arrows-alt')" size="16px" />
|
||||||
<q-tooltip :delay="1500" anchor="bottom middle" content-style="font-size: 80%">На весь экран</q-tooltip>
|
<q-tooltip :delay="1500" anchor="bottom middle" content-style="font-size: 80%">На весь экран</q-tooltip>
|
||||||
</span>
|
</span>
|
||||||
<span class="full-screen-button row justify-center items-center" @mousedown.stop @click="changeScale(0.1)">
|
<span class="header-button row justify-center items-center" @mousedown.stop @click="changeScale(0.1)">
|
||||||
<q-icon name="la la-plus" size="16px" />
|
<q-icon name="la la-plus" size="16px" />
|
||||||
<q-tooltip :delay="1500" anchor="bottom middle" content-style="font-size: 80%">Увеличить масштаб</q-tooltip>
|
<q-tooltip :delay="1500" anchor="bottom middle" content-style="font-size: 80%">Увеличить масштаб</q-tooltip>
|
||||||
</span>
|
</span>
|
||||||
<span class="full-screen-button row justify-center items-center" @mousedown.stop @click="changeScale(-0.1)">
|
<span class="header-button row justify-center items-center" @mousedown.stop @click="changeScale(-0.1)">
|
||||||
<q-icon name="la la-minus" size="16px" />
|
<q-icon name="la la-minus" size="16px" />
|
||||||
<q-tooltip :delay="1500" anchor="bottom middle" content-style="font-size: 80%">Уменьшить масштаб</q-tooltip>
|
<q-tooltip :delay="1500" anchor="bottom middle" content-style="font-size: 80%">Уменьшить масштаб</q-tooltip>
|
||||||
</span>
|
</span>
|
||||||
<span class="full-screen-button row justify-center items-center" @mousedown.stop @click="showHelp">
|
<span class="header-button row justify-center items-center" @mousedown.stop @click="showHelp">
|
||||||
<q-icon name="la la-question-circle" size="16px" />
|
<q-icon name="la la-question-circle" size="16px" />
|
||||||
<q-tooltip :delay="1500" anchor="bottom middle" content-style="font-size: 80%">Справка</q-tooltip>
|
<q-tooltip :delay="1500" anchor="bottom middle" content-style="font-size: 80%">Справка</q-tooltip>
|
||||||
</span>
|
</span>
|
||||||
@@ -32,7 +32,7 @@
|
|||||||
:options="rootLinkOptions"
|
:options="rootLinkOptions"
|
||||||
style="width: 230px"
|
style="width: 230px"
|
||||||
dropdown-icon="la la-angle-down la-sm"
|
dropdown-icon="la la-angle-down la-sm"
|
||||||
rounded outlined dense emit-value map-options display-value-sanitize options-sanitize
|
outlined dense emit-value map-options display-value-sanitize options-sanitize
|
||||||
@popup-show="onSelectPopupShow" @popup-hide="onSelectPopupHide"
|
@popup-show="onSelectPopupShow" @popup-hide="onSelectPopupHide"
|
||||||
>
|
>
|
||||||
<template #prepend>
|
<template #prepend>
|
||||||
@@ -61,7 +61,7 @@
|
|||||||
:options="selectedLinkOptions"
|
:options="selectedLinkOptions"
|
||||||
style="width: 50px"
|
style="width: 50px"
|
||||||
dropdown-icon="la la-angle-down la-sm"
|
dropdown-icon="la la-angle-down la-sm"
|
||||||
rounded outlined dense emit-value map-options hide-selected display-value-sanitize options-sanitize
|
outlined dense emit-value map-options hide-selected display-value-sanitize options-sanitize
|
||||||
@popup-show="onSelectPopupShow" @popup-hide="onSelectPopupHide"
|
@popup-show="onSelectPopupShow" @popup-hide="onSelectPopupHide"
|
||||||
>
|
>
|
||||||
<q-tooltip :delay="1500" anchor="bottom middle" content-style="font-size: 80%">
|
<q-tooltip :delay="1500" anchor="bottom middle" content-style="font-size: 80%">
|
||||||
@@ -73,9 +73,9 @@
|
|||||||
ref="input"
|
ref="input"
|
||||||
v-model="bookUrl"
|
v-model="bookUrl"
|
||||||
class="col q-mr-sm"
|
class="col q-mr-sm"
|
||||||
rounded outlined dense
|
outlined dense
|
||||||
bg-color="white"
|
bg-color="white"
|
||||||
placeholder="Скопируйте сюда URL книги"
|
placeholder="Скопируйте сюда ссылку на книгу и нажмите 'Открыть'"
|
||||||
@focus="selectAllOnFocus" @keydown="bookUrlKeyDown"
|
@focus="selectAllOnFocus" @keydown="bookUrlKeyDown"
|
||||||
>
|
>
|
||||||
<template #prepend>
|
<template #prepend>
|
||||||
@@ -99,7 +99,7 @@
|
|||||||
</template>
|
</template>
|
||||||
</q-input>
|
</q-input>
|
||||||
|
|
||||||
<q-btn :disabled="!bookUrl" rounded color="green-7" no-caps size="14px" @click="submitUrl">
|
<q-btn :disabled="!bookUrl" color="green-7" no-caps size="14px" @click="submitUrl">
|
||||||
Открыть
|
Открыть
|
||||||
<q-tooltip :delay="1500" anchor="bottom middle" content-style="font-size: 80%">
|
<q-tooltip :delay="1500" anchor="bottom middle" content-style="font-size: 80%">
|
||||||
Открыть в читалке
|
Открыть в читалке
|
||||||
@@ -894,14 +894,15 @@ export default vueComponent(ExternalLibs);
|
|||||||
background-color: #A0A0A0;
|
background-color: #A0A0A0;
|
||||||
}
|
}
|
||||||
|
|
||||||
.full-screen-button {
|
.header-button {
|
||||||
width: 30px;
|
width: 30px;
|
||||||
height: 30px;
|
height: 30px;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
}
|
}
|
||||||
|
|
||||||
.full-screen-button:hover {
|
.header-button:hover {
|
||||||
background-color: #69C05F;
|
color: white;
|
||||||
|
background-color: #39902F;
|
||||||
}
|
}
|
||||||
|
|
||||||
.transparent-layout {
|
.transparent-layout {
|
||||||
|
|||||||
@@ -23,15 +23,15 @@
|
|||||||
|
|
||||||
<div class="q-mb-sm" />
|
<div class="q-mb-sm" />
|
||||||
|
|
||||||
<div v-show="selectedTab == 'contents'" class="tab-panel">
|
<div v-show="selectedTab == 'contents'" ref="tabPanelContents" class="tab-panel">
|
||||||
<div>
|
<div>
|
||||||
<div v-for="item in contents" :key="item.key" class="column" style="width: 540px">
|
<div v-for="item in contents" :key="item.key" class="column" style="width: 540px">
|
||||||
<div class="row q-px-sm no-wrap" :class="{'item': !item.isBookPos, 'item-book-pos': item.isBookPos}">
|
<div :ref="`mainitem${item.key}`" class="row q-px-sm no-wrap" :class="{'item': !item.isBookPos, 'item-book-pos': item.isBookPos}">
|
||||||
<div v-if="item.list.length" class="row justify-center items-center expand-button clickable" @click="expandClick(item.key)">
|
<div v-if="item.list.length" class="row justify-center items-center expand-button clickable" @click="expandClick(item.key)">
|
||||||
<q-icon name="la la-caret-right" class="icon" :class="{'expanded-icon': item.expanded}" color="green-8" size="20px" />
|
<q-icon name="la la-caret-right" class="icon" :class="{'expanded-icon': item.expanded}" color="green-8" size="24px" />
|
||||||
</div>
|
</div>
|
||||||
<div v-else class="no-expand-button clickable" @click="setBookPos(item.offset)">
|
<div v-else class="no-expand-button clickable" @click="setBookPos(item.offset)">
|
||||||
<q-icon name="la la-stop" class="icon" style="visibility: hidden" size="20px" />
|
<q-icon name="la la-stop" class="icon" style="visibility: hidden" size="24px" />
|
||||||
</div>
|
</div>
|
||||||
<div class="col row clickable" @click="setBookPos(item.offset)">
|
<div class="col row clickable" @click="setBookPos(item.offset)">
|
||||||
<div :style="item.indentStyle"></div>
|
<div :style="item.indentStyle"></div>
|
||||||
@@ -42,8 +42,12 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div v-if="item.expanded" :ref="`subitem${item.key}`" class="subitems-transition">
|
<div v-if="item.expanded" :ref="`subdiv${item.key}`" class="subitems-transition">
|
||||||
<div v-for="subitem in item.list" :key="subitem.key" class="row q-px-sm no-wrap" :class="{'subitem': !subitem.isBookPos, 'subitem-book-pos': subitem.isBookPos}">
|
<div
|
||||||
|
v-for="subitem in item.list"
|
||||||
|
:ref="`subitem${subitem.key}`"
|
||||||
|
:key="subitem.key" class="row q-px-sm no-wrap" :class="{'subitem': !subitem.isBookPos, 'subitem-book-pos': subitem.isBookPos}"
|
||||||
|
>
|
||||||
<div class="col row clickable" @click="setBookPos(subitem.offset)">
|
<div class="col row clickable" @click="setBookPos(subitem.offset)">
|
||||||
<div class="no-expand-button"></div>
|
<div class="no-expand-button"></div>
|
||||||
<div :style="subitem.indentStyle"></div>
|
<div :style="subitem.indentStyle"></div>
|
||||||
@@ -61,10 +65,10 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div v-show="selectedTab == 'images'" class="tab-panel">
|
<div v-show="selectedTab == 'images'" ref="tabPanelImages" class="tab-panel">
|
||||||
<div>
|
<div>
|
||||||
<div v-for="item in images" :key="item.key" class="column" style="width: 540px">
|
<div v-for="item in images" :key="item.key" class="column" style="width: 540px">
|
||||||
<div class="row q-px-sm no-wrap" :class="{'item': !item.isBookPos, 'item-book-pos': item.isBookPos}">
|
<div :ref="`image${item.key}`" class="row q-px-sm no-wrap" :class="{'item': !item.isBookPos, 'item-book-pos': item.isBookPos}">
|
||||||
<div class="col row clickable" @click="setBookPos(item.offset)">
|
<div class="col row clickable" @click="setBookPos(item.offset)">
|
||||||
<div class="image-thumb-box row justify-center items-center">
|
<div class="image-thumb-box row justify-center items-center">
|
||||||
<div v-show="!imageLoaded[item.id]" class="image-thumb column justify-center">
|
<div v-show="!imageLoaded[item.id]" class="image-thumb column justify-center">
|
||||||
@@ -124,7 +128,10 @@ const componentOptions = {
|
|||||||
watch: {
|
watch: {
|
||||||
bookPos() {
|
bookPos() {
|
||||||
this.updateBookPosSelection();
|
this.updateBookPosSelection();
|
||||||
}
|
},
|
||||||
|
selectedTab() {
|
||||||
|
this.updateBookPosScrollTop();
|
||||||
|
},
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
class ContentsPage {
|
class ContentsPage {
|
||||||
@@ -282,31 +289,30 @@ class ContentsPage {
|
|||||||
if (!this.isVisible)
|
if (!this.isVisible)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
await utils.sleep(50);
|
await this.$nextTick();
|
||||||
const bp = this.bookPos;
|
const bp = this.bookPos;
|
||||||
|
|
||||||
for (let i = 0; i < this.contents.length; i++) {
|
for (let i = 0; i < this.contents.length; i++) {
|
||||||
const item = this.contents[i];
|
const item = this.contents[i];
|
||||||
const nextOffset = (i < this.contents.length - 1 ? this.contents[i + 1].offset : this.parsed.textLength);
|
const nextOffset = (i < this.contents.length - 1 ? this.contents[i + 1].offset : this.parsed.textLength);
|
||||||
|
|
||||||
|
if (bp >= item.offset && bp < nextOffset) {
|
||||||
|
item.isBookPos = true;
|
||||||
|
} else if (item.isBookPos) {
|
||||||
|
item.isBookPos = false;
|
||||||
|
}
|
||||||
|
|
||||||
for (let j = 0; j < item.list.length; j++) {
|
for (let j = 0; j < item.list.length; j++) {
|
||||||
const subitem = item.list[j];
|
const subitem = item.list[j];
|
||||||
const nextSubOffset = (j < item.list.length - 1 ? item.list[j + 1].offset : nextOffset);
|
const nextSubOffset = (j < item.list.length - 1 ? item.list[j + 1].offset : nextOffset);
|
||||||
|
|
||||||
if (bp >= subitem.offset && bp < nextSubOffset) {
|
if (bp >= subitem.offset && bp < nextSubOffset) {
|
||||||
subitem.isBookPos = true;
|
subitem.isBookPos = true;
|
||||||
this.contents[i] = Object.assign(item, {list: item.list});
|
this.updateBookPosScrollTop('contents', item, subitem, j);
|
||||||
} else if (subitem.isBookPos) {
|
} else if (subitem.isBookPos) {
|
||||||
subitem.isBookPos = false;
|
subitem.isBookPos = false;
|
||||||
this.contents[i] = Object.assign(item, {list: item.list});
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (bp >= item.offset && bp < nextOffset) {
|
|
||||||
this.contents[i] = Object.assign(item, {isBookPos: true});
|
|
||||||
} else if (item.isBookPos) {
|
|
||||||
this.contents[i] = Object.assign(item, {isBookPos: false});
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
for (let i = 0; i < this.images.length; i++) {
|
for (let i = 0; i < this.images.length; i++) {
|
||||||
@@ -314,11 +320,96 @@ class ContentsPage {
|
|||||||
const nextOffset = (i < this.images.length - 1 ? this.images[i + 1].offset : this.parsed.textLength);
|
const nextOffset = (i < this.images.length - 1 ? this.images[i + 1].offset : this.parsed.textLength);
|
||||||
|
|
||||||
if (bp >= img.offset && bp < nextOffset) {
|
if (bp >= img.offset && bp < nextOffset) {
|
||||||
this.images[i] = Object.assign(img, {isBookPos: true});
|
this.images[i].isBookPos = true;
|
||||||
} else if (img.isBookPos) {
|
} else if (img.isBookPos) {
|
||||||
this.images[i] = Object.assign(img, {isBookPos: false});
|
this.images[i].isBookPos = false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
this.updateBookPosScrollTop();
|
||||||
|
}
|
||||||
|
|
||||||
|
/*getOffsetTop(key) {
|
||||||
|
let el = this.getFirstElem(this.$refs[`mainitem${key}`]);
|
||||||
|
return (el ? el.offsetTop : 0);
|
||||||
|
}*/
|
||||||
|
|
||||||
|
async updateBookPosScrollTop() {
|
||||||
|
try {
|
||||||
|
await this.$nextTick();
|
||||||
|
|
||||||
|
if (this.selectedTab == 'contents') {
|
||||||
|
let item;
|
||||||
|
let subitem;
|
||||||
|
let i;
|
||||||
|
|
||||||
|
//ищем выделенные item
|
||||||
|
for(const _item of this.contents) {
|
||||||
|
if (_item.isBookPos) {
|
||||||
|
item = _item;
|
||||||
|
for (let ii = 0; ii < item.list.length; ii++) {
|
||||||
|
const _subitem = item.list[ii];
|
||||||
|
if (_subitem.isBookPos) {
|
||||||
|
subitem = _subitem;
|
||||||
|
i = ii;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!item)
|
||||||
|
return;
|
||||||
|
|
||||||
|
//вычисляем и смещаем tabPanel.scrollTop
|
||||||
|
let el = this.getFirstElem(this.$refs[`mainitem${item.key}`]);
|
||||||
|
let elShift = 0;
|
||||||
|
if (subitem && item.expanded) {
|
||||||
|
const subEl = this.getFirstElem(this.$refs[`subitem${subitem.key}`]);
|
||||||
|
elShift = el.offsetHeight - subEl.offsetHeight*(i + 1);
|
||||||
|
} else {
|
||||||
|
elShift = el.offsetHeight;
|
||||||
|
}
|
||||||
|
|
||||||
|
const tabPanel = this.$refs.tabPanelContents;
|
||||||
|
const halfH = tabPanel.clientHeight/2;
|
||||||
|
const newScrollTop = el.offsetTop - halfH - elShift;
|
||||||
|
if (newScrollTop < 20 + tabPanel.scrollTop - halfH || newScrollTop > -20 + tabPanel.scrollTop + halfH)
|
||||||
|
tabPanel.scrollTop = newScrollTop;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.selectedTab == 'images') {
|
||||||
|
let item;
|
||||||
|
|
||||||
|
//ищем выделенные item
|
||||||
|
for(const _item of this.images) {
|
||||||
|
if (_item.isBookPos) {
|
||||||
|
item = _item;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!item)
|
||||||
|
return;
|
||||||
|
|
||||||
|
//вычисляем и смещаем tabPanel.scrollTop
|
||||||
|
let el = this.getFirstElem(this.$refs[`image${item.key}`]);
|
||||||
|
|
||||||
|
const tabPanel = this.$refs.tabPanelImages;
|
||||||
|
const halfH = tabPanel.clientHeight/2;
|
||||||
|
const newScrollTop = el.offsetTop - halfH - el.offsetHeight/2;
|
||||||
|
|
||||||
|
if (newScrollTop < 20 + tabPanel.scrollTop - halfH || newScrollTop > -20 + tabPanel.scrollTop + halfH)
|
||||||
|
tabPanel.scrollTop = newScrollTop;
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
console.error(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
getFirstElem(items) {
|
||||||
|
return (Array.isArray(items) ? items[0] : items);
|
||||||
}
|
}
|
||||||
|
|
||||||
async expandClick(key) {
|
async expandClick(key) {
|
||||||
@@ -326,17 +417,17 @@ class ContentsPage {
|
|||||||
const expanded = !item.expanded;
|
const expanded = !item.expanded;
|
||||||
|
|
||||||
if (!expanded) {
|
if (!expanded) {
|
||||||
const subitems = this.$refs[`subitem${key}`];
|
let subdiv = this.getFirstElem(this.$refs[`subdiv${key}`]);
|
||||||
subitems.style.height = '0';
|
subdiv.style.height = '0';
|
||||||
await utils.sleep(200);
|
await utils.sleep(200);
|
||||||
}
|
}
|
||||||
|
|
||||||
this.contents[key] = Object.assign({}, item, {expanded});
|
this.contents[key].expanded = expanded;
|
||||||
|
|
||||||
if (expanded) {
|
if (expanded) {
|
||||||
await this.$nextTick();
|
await this.$nextTick();
|
||||||
const subitems = this.$refs[`subitem${key}`];
|
let subdiv = this.getFirstElem(this.$refs[`subdiv${key}`]);
|
||||||
subitems.style.height = subitems.scrollHeight + 'px';
|
subdiv.style.height = subdiv.scrollHeight + 'px';
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -19,7 +19,7 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="address">
|
<!--div class="address">
|
||||||
<img class="logo" src="./assets/paypal.png">
|
<img class="logo" src="./assets/paypal.png">
|
||||||
<div class="para">
|
<div class="para">
|
||||||
{{ paypalAddress }}
|
{{ paypalAddress }}
|
||||||
@@ -29,7 +29,7 @@
|
|||||||
</q-tooltip>
|
</q-tooltip>
|
||||||
</q-icon>
|
</q-icon>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div-->
|
||||||
|
|
||||||
<div class="address">
|
<div class="address">
|
||||||
<img class="logo" src="./assets/bitcoin.png">
|
<img class="logo" src="./assets/bitcoin.png">
|
||||||
|
|||||||
@@ -5,13 +5,20 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<div class="col column" style="min-width: 600px">
|
<div class="col column" style="min-width: 600px">
|
||||||
<q-btn-toggle
|
<div class="bg-grey-3 row">
|
||||||
v-model="selectedTab"
|
<q-tabs
|
||||||
toggle-color="primary"
|
v-model="selectedTab"
|
||||||
no-caps unelevated
|
active-color="black"
|
||||||
:options="buttons"
|
active-bg-color="white"
|
||||||
/>
|
indicator-color="white"
|
||||||
<div class="separator"></div>
|
dense
|
||||||
|
no-caps
|
||||||
|
inline-label
|
||||||
|
class="bg-grey-4 text-grey-7"
|
||||||
|
>
|
||||||
|
<q-tab v-for="btn in buttons" :key="btn.value" :name="btn.value" :label="btn.label" />
|
||||||
|
</q-tabs>
|
||||||
|
</div>
|
||||||
|
|
||||||
<keep-alive>
|
<keep-alive>
|
||||||
<component :is="activePage" ref="page" class="col"></component>
|
<component :is="activePage" ref="page" class="col"></component>
|
||||||
@@ -29,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 = [
|
||||||
@@ -44,7 +51,7 @@ const tabs = [
|
|||||||
['MouseHelpPage', 'Мышь/тачскрин'],
|
['MouseHelpPage', 'Мышь/тачскрин'],
|
||||||
['HotkeysHelpPage', 'Клавиатура'],
|
['HotkeysHelpPage', 'Клавиатура'],
|
||||||
['VersionHistoryPage', 'История версий'],
|
['VersionHistoryPage', 'История версий'],
|
||||||
['DonateHelpPage', 'Помочь проекту'],
|
//['DonateHelpPage', 'Помочь проекту'],
|
||||||
];
|
];
|
||||||
|
|
||||||
const componentOptions = {
|
const componentOptions = {
|
||||||
@@ -73,7 +80,7 @@ class HelpPage {
|
|||||||
}
|
}
|
||||||
|
|
||||||
activateDonateHelpPage() {
|
activateDonateHelpPage() {
|
||||||
this.selectedTab = 'DonateHelpPage';
|
//this.selectedTab = 'DonateHelpPage';
|
||||||
}
|
}
|
||||||
|
|
||||||
activateVersionHistoryHelpPage() {
|
activateVersionHistoryHelpPage() {
|
||||||
@@ -93,8 +100,4 @@ export default vueComponent(HelpPage);
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style scoped>
|
<style scoped>
|
||||||
.separator {
|
|
||||||
height: 1px;
|
|
||||||
background-color: #E0E0E0;
|
|
||||||
}
|
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -33,14 +33,15 @@ class VersionHistoryPage {
|
|||||||
|
|
||||||
mounted() {
|
mounted() {
|
||||||
let vh = [];
|
let vh = [];
|
||||||
for (const version of versionHistory) {
|
for (const v of versionHistory) {
|
||||||
vh.push(version.header);
|
vh.push(`${v.version} (${v.releaseDate})`);
|
||||||
}
|
}
|
||||||
this.versionHeader = vh;
|
this.versionHeader = vh;
|
||||||
|
|
||||||
let vc = [];
|
let vc = [];
|
||||||
for (const version of versionHistory) {
|
for (const v of versionHistory) {
|
||||||
vc.push({key: version.header, content: 'Версия ' + version.header + version.content});
|
let header = `${v.version} (${v.releaseDate})`;
|
||||||
|
vc.push({key: header, content: 'Версия ' + header + v.content});
|
||||||
}
|
}
|
||||||
this.versionContent = vc;
|
this.versionContent = vc;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -12,21 +12,31 @@
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="col-auto column justify-start items-center no-wrap overflow-hidden">
|
<div class="col-auto column justify-start items-center no-wrap overflow-hidden">
|
||||||
<q-input ref="input" v-model="bookUrl" class="full-width q-px-sm" style="max-width: 700px" outlined dense bg-color="white" placeholder="URL книги" @keydown="onInputKeydown">
|
<q-input
|
||||||
|
ref="input" v-model="bookUrl" class="full-width q-px-sm" style="max-width: 700px"
|
||||||
|
outlined dense bg-color="white" placeholder="Ссылка на книгу или веб-страницу" @keydown="onInputKeydown"
|
||||||
|
>
|
||||||
<template #append>
|
<template #append>
|
||||||
<q-btn rounded flat style="width: 40px" icon="la la-check" @click="submitUrl" />
|
<q-btn rounded flat style="width: 40px" icon="la la-check" @click="submitUrl" />
|
||||||
</template>
|
</template>
|
||||||
</q-input>
|
</q-input>
|
||||||
|
|
||||||
<input id="file" ref="file" type="file" style="display: none;" @change="loadFile" />
|
<input
|
||||||
|
id="file" ref="file" type="file"
|
||||||
|
style="display: none;"
|
||||||
|
:accept="acceptFileExt"
|
||||||
|
@change="loadFile"
|
||||||
|
/>
|
||||||
|
|
||||||
<div class="q-my-sm"></div>
|
<div class="q-my-sm"></div>
|
||||||
<q-btn no-caps dense class="q-px-sm" color="primary" size="13px" @click="loadFileClick">
|
<q-btn no-caps dense class="q-px-sm" color="primary" size="13px" @click="loadFileClick">
|
||||||
|
<q-icon class="q-mr-xs" name="la la-caret-square-up" size="24px" />
|
||||||
Загрузить файл с диска
|
Загрузить файл с диска
|
||||||
</q-btn>
|
</q-btn>
|
||||||
|
|
||||||
<div class="q-my-sm"></div>
|
<div class="q-my-sm"></div>
|
||||||
<q-btn no-caps dense class="q-px-sm" color="primary" size="13px" @click="loadBufferClick">
|
<q-btn no-caps dense class="q-px-sm" color="primary" size="13px" @click="loadBufferClick">
|
||||||
|
<q-icon class="q-mr-xs" name="la la-comment" size="24px" />
|
||||||
Из буфера обмена
|
Из буфера обмена
|
||||||
</q-btn>
|
</q-btn>
|
||||||
|
|
||||||
@@ -45,14 +55,27 @@
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<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 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>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<PasteTextPage v-if="pasteTextActive" ref="pasteTextPage" @paste-text-toggle="pasteTextToggle" @load-buffer="loadBuffer"></PasteTextPage>
|
<PasteTextPage v-if="pasteTextActive" ref="pasteTextPage" @paste-text-toggle="pasteTextToggle" @load-buffer="loadBuffer"></PasteTextPage>
|
||||||
|
|
||||||
|
<Dialog ref="dialog1" v-model="findBookVisible">
|
||||||
|
<template #header>
|
||||||
|
Подсказка ;-)
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<div style="word-break: normal">
|
||||||
|
Если вы хотите найти определенную книгу, добро пожаловать в
|
||||||
|
раздел "Сетевая библиотека" (кнопка <q-icon name="la la-sitemap" size="32px" />) на сайте читалки
|
||||||
|
<a href="https://liberama.top" target="_blank">liberama.top</a>
|
||||||
|
</div>
|
||||||
|
</Dialog>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
@@ -62,12 +85,15 @@ import vueComponent from '../../vueComponent.js';
|
|||||||
|
|
||||||
import GithubCorner from './GithubCorner/GithubCorner.vue';
|
import GithubCorner from './GithubCorner/GithubCorner.vue';
|
||||||
|
|
||||||
|
import Dialog from '../../share/Dialog.vue';
|
||||||
import PasteTextPage from './PasteTextPage/PasteTextPage.vue';
|
import PasteTextPage from './PasteTextPage/PasteTextPage.vue';
|
||||||
import {versionHistory} from '../versionHistory';
|
import {versionHistory} from '../versionHistory';
|
||||||
|
import * as utils from '../../../share/utils';
|
||||||
|
|
||||||
const componentOptions = {
|
const componentOptions = {
|
||||||
components: {
|
components: {
|
||||||
GithubCorner,
|
GithubCorner,
|
||||||
|
Dialog,
|
||||||
PasteTextPage,
|
PasteTextPage,
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
@@ -77,6 +103,7 @@ class LoaderPage {
|
|||||||
bookUrl = null;
|
bookUrl = null;
|
||||||
loadPercent = 0;
|
loadPercent = 0;
|
||||||
pasteTextActive = false;
|
pasteTextActive = false;
|
||||||
|
findBookVisible = false;
|
||||||
|
|
||||||
created() {
|
created() {
|
||||||
this.commit = this.$store.commit;
|
this.commit = this.$store.commit;
|
||||||
@@ -109,14 +136,16 @@ class LoaderPage {
|
|||||||
return this.$store.state.config.version;
|
return this.$store.state.config.version;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
get acceptFileExt() {
|
||||||
|
return this.$store.state.config.acceptFileExt;
|
||||||
|
}
|
||||||
|
|
||||||
get isExternalConverter() {
|
get isExternalConverter() {
|
||||||
return this.$store.state.config.useExternalBookConverter;
|
return this.$store.state.config.useExternalBookConverter;
|
||||||
}
|
}
|
||||||
|
|
||||||
get clientVersion() {
|
get clientVersion() {
|
||||||
let v = versionHistory[0].header;
|
return versionHistory[0].version;
|
||||||
v = v.split(' ')[0];
|
|
||||||
return v;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
submitUrl() {
|
submitUrl() {
|
||||||
@@ -138,7 +167,7 @@ class LoaderPage {
|
|||||||
}
|
}
|
||||||
|
|
||||||
loadBufferClick() {
|
loadBufferClick() {
|
||||||
this.pasteTextToggle();
|
this.showPasteText();
|
||||||
}
|
}
|
||||||
|
|
||||||
loadBuffer(opts) {
|
loadBuffer(opts) {
|
||||||
@@ -148,6 +177,10 @@ class LoaderPage {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
showPasteText() {
|
||||||
|
this.pasteTextActive = true;
|
||||||
|
}
|
||||||
|
|
||||||
pasteTextToggle() {
|
pasteTextToggle() {
|
||||||
this.pasteTextActive = !this.pasteTextActive;
|
this.pasteTextActive = !this.pasteTextActive;
|
||||||
}
|
}
|
||||||
@@ -160,6 +193,10 @@ class LoaderPage {
|
|||||||
this.$emit('do-action', {action: 'donate'});
|
this.$emit('do-action', {action: 'donate'});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
findBook() {
|
||||||
|
this.findBookVisible = true;
|
||||||
|
}
|
||||||
|
|
||||||
openComments() {
|
openComments() {
|
||||||
window.open('http://samlib.ru/comment/b/bookpauk/bookpauk_reader', '_blank');
|
window.open('http://samlib.ru/comment/b/bookpauk/bookpauk_reader', '_blank');
|
||||||
}
|
}
|
||||||
@@ -168,26 +205,24 @@ class LoaderPage {
|
|||||||
window.open('http://old.omnireader.ru', '_blank');
|
window.open('http://old.omnireader.ru', '_blank');
|
||||||
}
|
}
|
||||||
|
|
||||||
onInputKeydown(event) {
|
async onInputKeydown(event) {
|
||||||
if (event.key == 'Enter') {
|
if (event.key == 'Enter') {
|
||||||
|
await utils.sleep(100);
|
||||||
this.submitUrl();
|
this.submitUrl();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
keyHook(event) {
|
keyHook(event) {
|
||||||
|
if (this.$refs.dialog1.active)
|
||||||
|
return true;
|
||||||
|
|
||||||
if (this.pasteTextActive) {
|
if (this.pasteTextActive) {
|
||||||
return this.$refs.pasteTextPage.keyHook(event);
|
return this.$refs.pasteTextPage.keyHook(event);
|
||||||
}
|
}
|
||||||
|
|
||||||
const input = this.$refs.input.getNativeElement();
|
const input = this.$refs.input.getNativeElement();
|
||||||
if (event.type == 'keydown' && document.activeElement !== input) {
|
if (event.type == 'keydown' && (document.activeElement === input || event.code == 'Enter') && event.code != 'Escape')
|
||||||
const action = this.$root.readerActionByKeyEvent(event);
|
return true;
|
||||||
switch (action) {
|
|
||||||
case 'help':
|
|
||||||
this.openHelp(event);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
<template>
|
<template>
|
||||||
<div v-show="visible" class="column justify-center items-center z-max" style="background-color: rgba(0, 0, 0, 0.8)">
|
<div v-show="visible" class="column justify-center items-center" style="background-color: rgba(0, 0, 0, 0.8); z-index: 100;">
|
||||||
<div class="column justify-start items-center" style="height: 250px">
|
<div class="column justify-start items-center" style="height: 250px">
|
||||||
<q-circular-progress
|
<q-circular-progress
|
||||||
show-value
|
show-value
|
||||||
|
|||||||
@@ -2,16 +2,35 @@
|
|||||||
<div class="column no-wrap">
|
<div class="column no-wrap">
|
||||||
<div v-show="toolBarActive" ref="header" class="header">
|
<div v-show="toolBarActive" ref="header" class="header">
|
||||||
<div ref="buttons" class="row justify-between no-wrap">
|
<div ref="buttons" class="row justify-between no-wrap">
|
||||||
<div>
|
<div class="row no-wrap">
|
||||||
<button ref="loader" v-ripple class="tool-button" :class="buttonActiveClass('loader')" @click="buttonClick('loader')">
|
<button ref="loader" v-ripple class="tool-button" :class="buttonActiveClass('loader')" @click="buttonClick('loader')">
|
||||||
<q-icon name="la la-arrow-left" size="32px" />
|
<q-icon name="la la-arrow-left" size="32px" />
|
||||||
<q-tooltip :delay="1500" anchor="bottom right" content-style="font-size: 80%">
|
<q-tooltip :delay="1500" anchor="bottom right" content-style="font-size: 80%">
|
||||||
{{ rstore.readerActions['loader'] }}
|
{{ rstore.readerActions['loader'] }}
|
||||||
</q-tooltip>
|
</q-tooltip>
|
||||||
</button>
|
</button>
|
||||||
|
<button v-show="showToolButton['loadFile']" ref="loadFile" v-ripple class="tool-button" :class="buttonActiveClass('loadFile')" @click="buttonClick('loadFile')">
|
||||||
|
<q-icon name="la la-caret-square-up" size="32px" />
|
||||||
|
<q-tooltip :delay="1500" anchor="bottom right" content-style="font-size: 80%">
|
||||||
|
{{ rstore.readerActions['loadFile'] }}
|
||||||
|
</q-tooltip>
|
||||||
|
</button>
|
||||||
|
<button v-show="showToolButton['loadBuffer']" ref="loadBuffer" v-ripple class="tool-button" :class="buttonActiveClass('loadBuffer')" @click="buttonClick('loadBuffer')">
|
||||||
|
<q-icon name="la la-comment" size="32px" />
|
||||||
|
<q-tooltip :delay="1500" anchor="bottom right" content-style="font-size: 80%">
|
||||||
|
{{ rstore.readerActions['loadBuffer'] }}
|
||||||
|
</q-tooltip>
|
||||||
|
</button>
|
||||||
|
<button v-show="showToolButton['help']" ref="help" v-ripple class="tool-button" :class="buttonActiveClass('help')" @click="buttonClick('help')">
|
||||||
|
<q-icon name="la la-question" size="32px" />
|
||||||
|
<q-tooltip :delay="1500" anchor="bottom right" content-style="font-size: 80%">
|
||||||
|
{{ rstore.readerActions['help'] }}
|
||||||
|
</q-tooltip>
|
||||||
|
</button>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div>
|
<div class="row no-wrap">
|
||||||
|
<div class="space"></div>
|
||||||
<button v-show="showToolButton['undoAction']" ref="undoAction" v-ripple class="tool-button" :class="buttonActiveClass('undoAction')" @click="buttonClick('undoAction')">
|
<button v-show="showToolButton['undoAction']" ref="undoAction" v-ripple class="tool-button" :class="buttonActiveClass('undoAction')" @click="buttonClick('undoAction')">
|
||||||
<q-icon name="la la-angle-left" size="32px" />
|
<q-icon name="la la-angle-left" 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%">
|
||||||
@@ -86,9 +105,16 @@
|
|||||||
{{ rstore.readerActions['recentBooks'] }}
|
{{ rstore.readerActions['recentBooks'] }}
|
||||||
</q-tooltip>
|
</q-tooltip>
|
||||||
</button>
|
</button>
|
||||||
|
<div class="space"></div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div>
|
<div class="row no-wrap">
|
||||||
|
<button v-show="showToolButton['clickControl']" ref="clickControl" v-ripple class="tool-button" :class="buttonActiveClass('clickControl')" @click="buttonClick('clickControl')">
|
||||||
|
<q-icon name="la la-mouse" size="32px" />
|
||||||
|
<q-tooltip :delay="1500" anchor="bottom middle" content-style="font-size: 80%">
|
||||||
|
{{ rstore.readerActions['clickControl'] }}
|
||||||
|
</q-tooltip>
|
||||||
|
</button>
|
||||||
<button v-show="showToolButton['offlineMode']" ref="offlineMode" v-ripple class="tool-button" :class="buttonActiveClass('offlineMode')" @click="buttonClick('offlineMode')">
|
<button v-show="showToolButton['offlineMode']" ref="offlineMode" v-ripple class="tool-button" :class="buttonActiveClass('offlineMode')" @click="buttonClick('offlineMode')">
|
||||||
<q-icon name="la la-unlink" size="32px" />
|
<q-icon name="la la-unlink" 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%">
|
||||||
@@ -115,6 +141,7 @@
|
|||||||
@load-file="loadFile"
|
@load-file="loadFile"
|
||||||
@book-pos-changed="bookPosChanged"
|
@book-pos-changed="bookPosChanged"
|
||||||
@do-action="doAction"
|
@do-action="doAction"
|
||||||
|
@hide-tool-bar="hideToolBar"
|
||||||
></component>
|
></component>
|
||||||
</keep-alive>
|
</keep-alive>
|
||||||
|
|
||||||
@@ -136,7 +163,7 @@
|
|||||||
<ContentsPage v-show="contentsActive" ref="contentsPage" :book-pos="bookPos" :is-visible="contentsActive" @do-action="doAction" @book-pos-changed="bookPosChanged"></ContentsPage>
|
<ContentsPage v-show="contentsActive" ref="contentsPage" :book-pos="bookPos" :is-visible="contentsActive" @do-action="doAction" @book-pos-changed="bookPosChanged"></ContentsPage>
|
||||||
|
|
||||||
<ServerStorage v-show="hidden" ref="serverStorage"></ServerStorage>
|
<ServerStorage v-show="hidden" ref="serverStorage"></ServerStorage>
|
||||||
<ReaderDialogs ref="dialogs" @donate-toggle="donateToggle" @version-history-toggle="versionHistoryToggle"></ReaderDialogs>
|
<ReaderDialogs ref="dialogs" @donate-toggle="donateToggle" @version-history-toggle="versionHistoryToggle" @load-buffer-toggle="loadBufferToggle"></ReaderDialogs>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
@@ -167,6 +194,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';
|
||||||
@@ -175,6 +203,7 @@ import miscApi from '../../api/misc';
|
|||||||
|
|
||||||
import {versionHistory} from './versionHistory';
|
import {versionHistory} from './versionHistory';
|
||||||
import * as utils from '../../share/utils';
|
import * as utils from '../../share/utils';
|
||||||
|
import LockQueue from '../../share/LockQueue';
|
||||||
|
|
||||||
const componentOptions = {
|
const componentOptions = {
|
||||||
components: {
|
components: {
|
||||||
@@ -245,6 +274,8 @@ class Reader {
|
|||||||
rstore = {};
|
rstore = {};
|
||||||
|
|
||||||
loaderActive = false;
|
loaderActive = false;
|
||||||
|
loadFileActive = false;
|
||||||
|
loadBufferActive = false;
|
||||||
fullScreenActive = false;
|
fullScreenActive = false;
|
||||||
setPositionActive = false;
|
setPositionActive = false;
|
||||||
searchActive = false;
|
searchActive = false;
|
||||||
@@ -254,6 +285,7 @@ class Reader {
|
|||||||
contentsActive = false;
|
contentsActive = false;
|
||||||
libsActive = false;
|
libsActive = false;
|
||||||
recentBooksActive = false;
|
recentBooksActive = false;
|
||||||
|
clickControlActive = false;
|
||||||
offlineModeActive = false;
|
offlineModeActive = false;
|
||||||
settingsActive = false;
|
settingsActive = false;
|
||||||
|
|
||||||
@@ -284,6 +316,8 @@ class Reader {
|
|||||||
this.reader = this.$store.state.reader;
|
this.reader = this.$store.state.reader;
|
||||||
this.config = this.$store.state.config;
|
this.config = this.$store.state.config;
|
||||||
|
|
||||||
|
this.lock = new LockQueue(100);
|
||||||
|
|
||||||
this.$root.addEventHook('key', this.keyHook);
|
this.$root.addEventHook('key', this.keyHook);
|
||||||
|
|
||||||
this.lastActivePage = false;
|
this.lastActivePage = false;
|
||||||
@@ -310,12 +344,19 @@ class Reader {
|
|||||||
await this.$nextTick();
|
await this.$nextTick();
|
||||||
this.paramPosIgnore = false;
|
this.paramPosIgnore = false;
|
||||||
}
|
}
|
||||||
}, 500, {maxWait: 5000});
|
}, 250, {maxWait: 5000});
|
||||||
|
|
||||||
this.scrollingSetRecentBook = _.debounce((newValue) => {
|
this.scrollingSetRecentBook = _.debounce((newValue) => {
|
||||||
this.debouncedSetRecentBook(newValue);
|
this.debouncedSetRecentBook(newValue);
|
||||||
}, 15000, {maxWait: 20000});
|
}, 15000, {maxWait: 20000});
|
||||||
|
|
||||||
|
this.debouncedHideToolBar = _.debounce((event) => {
|
||||||
|
if (this.toolBarHideOnScroll && this.toolBarActive !== !!event.show) {
|
||||||
|
this.commit('reader/setToolBarActive', !!event.show);
|
||||||
|
this.$root.eventHook('resize');
|
||||||
|
}
|
||||||
|
}, 200);
|
||||||
|
|
||||||
document.addEventListener('fullscreenchange', () => {
|
document.addEventListener('fullscreenchange', () => {
|
||||||
this.fullScreenActive = (document.fullscreenElement !== null);
|
this.fullScreenActive = (document.fullscreenElement !== null);
|
||||||
});
|
});
|
||||||
@@ -324,10 +365,10 @@ class Reader {
|
|||||||
}
|
}
|
||||||
|
|
||||||
mounted() {
|
mounted() {
|
||||||
this.updateHeaderMinWidth();
|
|
||||||
|
|
||||||
(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);
|
||||||
|
|
||||||
@@ -372,8 +413,10 @@ class Reader {
|
|||||||
this.copyFullText = settings.copyFullText;
|
this.copyFullText = settings.copyFullText;
|
||||||
this.showClickMapPage = settings.showClickMapPage;
|
this.showClickMapPage = settings.showClickMapPage;
|
||||||
this.clickControl = settings.clickControl;
|
this.clickControl = settings.clickControl;
|
||||||
|
this.clickControlActive = this.clickControl;
|
||||||
this.blinkCachedLoad = settings.blinkCachedLoad;
|
this.blinkCachedLoad = settings.blinkCachedLoad;
|
||||||
this.showToolButton = settings.showToolButton;
|
this.showToolButton = settings.showToolButton;
|
||||||
|
this.toolBarHideOnScroll = settings.toolBarHideOnScroll;
|
||||||
this.enableSitesFilter = settings.enableSitesFilter;
|
this.enableSitesFilter = settings.enableSitesFilter;
|
||||||
this.showNeedUpdateNotify = settings.showNeedUpdateNotify;
|
this.showNeedUpdateNotify = settings.showNeedUpdateNotify;
|
||||||
this.splitToPara = settings.splitToPara;
|
this.splitToPara = settings.splitToPara;
|
||||||
@@ -388,29 +431,69 @@ class Reader {
|
|||||||
return this.readerActionByKeyCode[utils.keyEventToCode(event)];
|
return this.readerActionByKeyCode[utils.keyEventToCode(event)];
|
||||||
}
|
}
|
||||||
|
|
||||||
this.updateHeaderMinWidth();
|
|
||||||
|
|
||||||
this.loadWallpapers();//no await
|
this.loadWallpapers();//no await
|
||||||
}
|
}
|
||||||
|
|
||||||
|
showHelpOnErrorIfNeeded(errorMessage) {
|
||||||
|
//небольшая эвристика
|
||||||
|
let i = errorMessage.indexOf('http://');
|
||||||
|
if (i < 0)
|
||||||
|
i = errorMessage.indexOf('https://');
|
||||||
|
|
||||||
|
errorMessage = errorMessage.substring(i + 7);
|
||||||
|
const perCount = errorMessage.split('%').length - 1;
|
||||||
|
|
||||||
|
if (perCount > errorMessage.length/3.2) {
|
||||||
|
this.$refs.dialogs.showUrlHelp();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
//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);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -439,17 +522,6 @@ class Reader {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
updateHeaderMinWidth() {
|
|
||||||
const showButtonCount = Object.values(this.showToolButton).reduce((a, b) => a + (b ? 1 : 0), 0);
|
|
||||||
if (this.$refs.buttons)
|
|
||||||
this.$refs.buttons.style.minWidth = 65*showButtonCount + 'px';
|
|
||||||
(async() => {
|
|
||||||
await utils.sleep(1000);
|
|
||||||
if (this.$refs.header)
|
|
||||||
this.$refs.header.style.overflowX = 'auto';
|
|
||||||
})();
|
|
||||||
}
|
|
||||||
|
|
||||||
checkSetStorageAccessKey() {
|
checkSetStorageAccessKey() {
|
||||||
const q = this.$route.query;
|
const q = this.$route.query;
|
||||||
|
|
||||||
@@ -525,9 +597,7 @@ class Reader {
|
|||||||
}
|
}
|
||||||
|
|
||||||
get clientVersion() {
|
get clientVersion() {
|
||||||
let v = versionHistory[0].header;
|
return versionHistory[0].version;
|
||||||
v = v.split(' ')[0];
|
|
||||||
return v;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
get routeParamUrl() {
|
get routeParamUrl() {
|
||||||
@@ -585,7 +655,20 @@ class Reader {
|
|||||||
//сохранение в serverStorage
|
//сохранение в serverStorage
|
||||||
if (value) {
|
if (value) {
|
||||||
await utils.sleep(500);
|
await utils.sleep(500);
|
||||||
await this.$refs.serverStorage.saveRecent(value);
|
|
||||||
|
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);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -619,6 +702,10 @@ class Reader {
|
|||||||
this.$root.eventHook('resize');
|
this.$root.eventHook('resize');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
hideToolBar(event) {
|
||||||
|
this.debouncedHideToolBar(event);
|
||||||
|
}
|
||||||
|
|
||||||
fullScreenToggle() {
|
fullScreenToggle() {
|
||||||
this.fullScreenActive = !this.fullScreenActive;
|
this.fullScreenActive = !this.fullScreenActive;
|
||||||
if (this.fullScreenActive) {
|
if (this.fullScreenActive) {
|
||||||
@@ -646,6 +733,28 @@ class Reader {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
loadFileToggle() {
|
||||||
|
if (!this.loaderActive)
|
||||||
|
this.loaderToggle();
|
||||||
|
this.$nextTick(() => {
|
||||||
|
const page = this.$refs.page;
|
||||||
|
if (this.activePage == 'LoaderPage' && page.loadFileClick) {
|
||||||
|
page.loadFileClick();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
loadBufferToggle() {
|
||||||
|
if (!this.loaderActive)
|
||||||
|
this.loaderToggle();
|
||||||
|
this.$nextTick(() => {
|
||||||
|
const page = this.$refs.page;
|
||||||
|
if (this.activePage == 'LoaderPage' && page.showPasteText) {
|
||||||
|
page.showPasteText();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
setPositionToggle() {
|
setPositionToggle() {
|
||||||
this.setPositionActive = !this.setPositionActive;
|
this.setPositionActive = !this.setPositionActive;
|
||||||
const page = this.$refs.page;
|
const page = this.$refs.page;
|
||||||
@@ -773,6 +882,12 @@ class Reader {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
clickControlToggle() {
|
||||||
|
const newSettings = _.cloneDeep(this.settings);
|
||||||
|
newSettings.clickControl = !this.clickControl;
|
||||||
|
this.commit('reader/setSettings', newSettings);
|
||||||
|
}
|
||||||
|
|
||||||
offlineModeToggle() {
|
offlineModeToggle() {
|
||||||
this.offlineModeActive = !this.offlineModeActive;
|
this.offlineModeActive = !this.offlineModeActive;
|
||||||
this.$refs.serverStorage.offlineModeActive = this.offlineModeActive;
|
this.$refs.serverStorage.offlineModeActive = this.offlineModeActive;
|
||||||
@@ -826,7 +941,7 @@ class Reader {
|
|||||||
|
|
||||||
refreshBook() {
|
refreshBook() {
|
||||||
const mrb = this.mostRecentBook();
|
const mrb = this.mostRecentBook();
|
||||||
this.loadBook({url: mrb.url, uploadFileName: mrb.uploadFileName, force: true});
|
this.loadBook(Object.assign({}, mrb, {force: true}));
|
||||||
}
|
}
|
||||||
|
|
||||||
undoAction() {
|
undoAction() {
|
||||||
@@ -861,6 +976,9 @@ class Reader {
|
|||||||
|
|
||||||
switch (action) {
|
switch (action) {
|
||||||
case 'loader':
|
case 'loader':
|
||||||
|
case 'loadFile':
|
||||||
|
case 'loadBuffer':
|
||||||
|
case 'help':
|
||||||
case 'fullScreen':
|
case 'fullScreen':
|
||||||
case 'setPosition':
|
case 'setPosition':
|
||||||
case 'search':
|
case 'search':
|
||||||
@@ -870,6 +988,7 @@ class Reader {
|
|||||||
case 'contents':
|
case 'contents':
|
||||||
case 'libs':
|
case 'libs':
|
||||||
case 'recentBooks':
|
case 'recentBooks':
|
||||||
|
case 'clickControl':
|
||||||
case 'offlineMode':
|
case 'offlineMode':
|
||||||
case 'settings':
|
case 'settings':
|
||||||
if (this.progressActive) {
|
if (this.progressActive) {
|
||||||
@@ -907,7 +1026,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;
|
||||||
@@ -976,7 +1094,7 @@ class Reader {
|
|||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
async loadBook(opts) {
|
async _loadBook(opts) {
|
||||||
if (!opts || !opts.url) {
|
if (!opts || !opts.url) {
|
||||||
this.mostRecentBook();
|
this.mostRecentBook();
|
||||||
return;
|
return;
|
||||||
@@ -986,10 +1104,6 @@ class Reader {
|
|||||||
|
|
||||||
let url = encodeURI(decodeURI(opts.url));
|
let url = encodeURI(decodeURI(opts.url));
|
||||||
|
|
||||||
//TODO: убрать конвертирование 'file://' после 06.2021
|
|
||||||
if (url.length == 71 && url.indexOf('file://') == 0)
|
|
||||||
url = url.replace(/^file/, 'disk');
|
|
||||||
|
|
||||||
if ((url.indexOf('http://') != 0) && (url.indexOf('https://') != 0) &&
|
if ((url.indexOf('http://') != 0) && (url.indexOf('https://') != 0) &&
|
||||||
(url.indexOf('disk://') != 0))
|
(url.indexOf('disk://') != 0))
|
||||||
url = 'http://' + url;
|
url = 'http://' + url;
|
||||||
@@ -1016,33 +1130,37 @@ class Reader {
|
|||||||
progress.show();
|
progress.show();
|
||||||
progress.setState({state: 'parse'});
|
progress.setState({state: 'parse'});
|
||||||
|
|
||||||
// есть ли среди недавних
|
// есть ли среди загруженных
|
||||||
const key = bookManager.keyFromUrl(url);
|
let wasOpened = bookManager.findRecentByUrlAndPath(url, opts.path);
|
||||||
let wasOpened = await bookManager.getRecentBook({key});
|
wasOpened = (wasOpened ? _.cloneDeep(wasOpened) : {});
|
||||||
wasOpened = (wasOpened ? wasOpened : {});
|
|
||||||
const bookPos = (opts.bookPos !== undefined ? opts.bookPos : wasOpened.bookPos);
|
wasOpened = Object.assign(wasOpened, {
|
||||||
const bookPosSeen = (opts.bookPos !== undefined ? opts.bookPos : wasOpened.bookPosSeen);
|
url: (opts.url !== undefined ? opts.url : wasOpened.url),
|
||||||
const uploadFileName = (opts.uploadFileName ? opts.uploadFileName : '');
|
path: (opts.path !== undefined ? opts.path : wasOpened.path),
|
||||||
|
bookPos: (opts.bookPos !== undefined ? opts.bookPos : wasOpened.bookPos),
|
||||||
|
bookPosSeen: (opts.bookPos !== undefined ? opts.bookPos : wasOpened.bookPosSeen),
|
||||||
|
uploadFileName: (opts.uploadFileName ? opts.uploadFileName : wasOpened.uploadFileName),
|
||||||
|
});
|
||||||
|
|
||||||
let book = null;
|
let book = null;
|
||||||
|
|
||||||
if (!opts.force) {
|
if (!opts.force) {
|
||||||
// пытаемся загрузить и распарсить книгу в менеджере из локального кэша
|
// пытаемся загрузить и распарсить книгу в менеджере из локального кэша
|
||||||
const bookParsed = await bookManager.getBook({url, path: opts.path}, (prog) => {
|
const bookParsed = await bookManager.getBook(wasOpened, (prog) => {
|
||||||
progress.setState({progress: prog});
|
progress.setState({progress: prog});
|
||||||
});
|
});
|
||||||
|
|
||||||
// если есть в локальном кэше
|
// если есть в локальном кэше
|
||||||
if (bookParsed) {
|
if (bookParsed) {
|
||||||
await bookManager.setRecentBook(Object.assign({bookPos, bookPosSeen}, bookParsed));
|
await bookManager.setRecentBook(Object.assign(wasOpened, bookParsed));
|
||||||
this.mostRecentBook();
|
this.mostRecentBook();
|
||||||
this.addAction(bookPos);
|
this.addAction(wasOpened.bookPos);
|
||||||
this.loaderActive = false;
|
this.loaderActive = false;
|
||||||
progress.hide(); this.progressActive = false;
|
progress.hide(); this.progressActive = false;
|
||||||
this.blinkCachedLoadMessage();
|
this.blinkCachedLoadMessage();
|
||||||
|
|
||||||
this.checkBookPosPercent();
|
this.checkBookPosPercent();
|
||||||
await this.activateClickMapPage();
|
this.activateClickMapPage();//no await
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1056,7 +1174,7 @@ class Reader {
|
|||||||
});
|
});
|
||||||
book = Object.assign({}, wasOpened, {data: resp.data});
|
book = Object.assign({}, wasOpened, {data: resp.data});
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
//молчим
|
this.$root.notify.error('Конвертированный файл не найден на сервере.<br>Пробуем загрузить оригинал.', 'Ошибка загрузки');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -1067,7 +1185,7 @@ class Reader {
|
|||||||
if (!book) {
|
if (!book) {
|
||||||
book = await readerApi.loadBook({
|
book = await readerApi.loadBook({
|
||||||
url,
|
url,
|
||||||
uploadFileName,
|
uploadFileName: wasOpened.uploadFileName,
|
||||||
enableSitesFilter: this.enableSitesFilter,
|
enableSitesFilter: this.enableSitesFilter,
|
||||||
skipHtmlCheck: (this.splitToPara ? true : false),
|
skipHtmlCheck: (this.splitToPara ? true : false),
|
||||||
isText: (this.splitToPara ? true : false),
|
isText: (this.splitToPara ? true : false),
|
||||||
@@ -1084,14 +1202,44 @@ class Reader {
|
|||||||
|
|
||||||
// добавляем в bookManager
|
// добавляем в bookManager
|
||||||
progress.setState({state: 'parse', step: 5});
|
progress.setState({state: 'parse', step: 5});
|
||||||
|
|
||||||
const addedBook = await bookManager.addBook(book, (prog) => {
|
const addedBook = await bookManager.addBook(book, (prog) => {
|
||||||
progress.setState({progress: prog});
|
progress.setState({progress: prog});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// sameBookKey
|
||||||
|
if (url.indexOf('disk://') == 0) {
|
||||||
|
//ищем такой файл в загруженных
|
||||||
|
let found = bookManager.findRecentBySameBookKey(wasOpened.uploadFileName);
|
||||||
|
found = (found ? _.cloneDeep(found) : found);
|
||||||
|
|
||||||
|
if (found) {
|
||||||
|
if (wasOpened.sameBookKey != found.sameBookKey) {
|
||||||
|
//спрашиваем, надо ли объединить файлы
|
||||||
|
const askResult = bookManager.keysEqual(found.path, addedBook.path) ||
|
||||||
|
await this.$root.stdDialog.askYesNo(`
|
||||||
|
Файл с именем "${wasOpened.uploadFileName}" уже есть в загруженных.
|
||||||
|
<br>Объединить позицию?`, 'Найдена похожая книга');
|
||||||
|
if (askResult) {
|
||||||
|
wasOpened.bookPos = found.bookPos;
|
||||||
|
wasOpened.bookPosSeen = found.bookPosSeen;
|
||||||
|
wasOpened.sameBookKey = found.sameBookKey;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
wasOpened.sameBookKey = wasOpened.uploadFileName;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
wasOpened.sameBookKey = addedBook.url;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!bookManager.keysEqual(wasOpened.path, addedBook.path))
|
||||||
|
delete wasOpened.loadTime;
|
||||||
|
|
||||||
// добавляем в историю
|
// добавляем в историю
|
||||||
await bookManager.setRecentBook(Object.assign({bookPos, bookPosSeen, uploadFileName}, addedBook));
|
await bookManager.setRecentBook(Object.assign(wasOpened, addedBook));
|
||||||
this.mostRecentBook();
|
this.mostRecentBook();
|
||||||
this.addAction(bookPos);
|
this.addAction(wasOpened.bookPos);
|
||||||
this.updateRoute(true);
|
this.updateRoute(true);
|
||||||
|
|
||||||
this.loaderActive = false;
|
this.loaderActive = false;
|
||||||
@@ -1102,17 +1250,28 @@ class Reader {
|
|||||||
this.stopBlink = true;
|
this.stopBlink = true;
|
||||||
|
|
||||||
this.checkBookPosPercent();
|
this.checkBookPosPercent();
|
||||||
await this.activateClickMapPage();
|
this.activateClickMapPage();//no await
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
progress.hide(); this.progressActive = false;
|
progress.hide(); this.progressActive = false;
|
||||||
this.loaderActive = true;
|
this.loaderActive = true;
|
||||||
this.$root.stdDialog.alert(e.message, 'Ошибка', {color: 'negative'});
|
if (!this.showHelpOnErrorIfNeeded(url)) {
|
||||||
|
this.$root.stdDialog.alert(e.message, 'Ошибка', {color: 'negative'});
|
||||||
|
}
|
||||||
} finally {
|
} finally {
|
||||||
this.checkNewVersionAvailable();
|
this.checkNewVersionAvailable();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async loadFile(opts) {
|
async loadBook(opts) {
|
||||||
|
await this.lock.get();
|
||||||
|
try {
|
||||||
|
await this._loadBook(opts);
|
||||||
|
} finally {
|
||||||
|
this.lock.ret();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async _loadFile(opts) {
|
||||||
this.progressActive = true;
|
this.progressActive = true;
|
||||||
|
|
||||||
await this.$nextTick();
|
await this.$nextTick();
|
||||||
@@ -1128,7 +1287,7 @@ class Reader {
|
|||||||
|
|
||||||
progress.hide(); this.progressActive = false;
|
progress.hide(); this.progressActive = false;
|
||||||
|
|
||||||
await this.loadBook({url, uploadFileName: opts.file.name, force: true});
|
await this._loadBook({url, uploadFileName: opts.file.name, force: true});
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
progress.hide(); this.progressActive = false;
|
progress.hide(); this.progressActive = false;
|
||||||
this.loaderActive = true;
|
this.loaderActive = true;
|
||||||
@@ -1136,6 +1295,15 @@ class Reader {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async loadFile(opts) {
|
||||||
|
await this.lock.get();
|
||||||
|
try {
|
||||||
|
await this._loadFile(opts);
|
||||||
|
} finally {
|
||||||
|
this.lock.ret();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
blinkCachedLoadMessage() {
|
blinkCachedLoadMessage() {
|
||||||
if (!this.blinkCachedLoad)
|
if (!this.blinkCachedLoad)
|
||||||
return;
|
return;
|
||||||
@@ -1172,6 +1340,12 @@ class Reader {
|
|||||||
case 'loader':
|
case 'loader':
|
||||||
this.loaderToggle();
|
this.loaderToggle();
|
||||||
break;
|
break;
|
||||||
|
case 'loadFile':
|
||||||
|
this.loadFileToggle();
|
||||||
|
break;
|
||||||
|
case 'loadBuffer':
|
||||||
|
this.loadBufferToggle();
|
||||||
|
break;
|
||||||
case 'help':
|
case 'help':
|
||||||
this.helpToggle();
|
this.helpToggle();
|
||||||
break;
|
break;
|
||||||
@@ -1214,6 +1388,9 @@ class Reader {
|
|||||||
case 'recentBooks':
|
case 'recentBooks':
|
||||||
this.recentBooksToggle();
|
this.recentBooksToggle();
|
||||||
break;
|
break;
|
||||||
|
case 'clickControl':
|
||||||
|
this.clickControlToggle();
|
||||||
|
break;
|
||||||
case 'offlineMode':
|
case 'offlineMode':
|
||||||
this.offlineModeToggle();
|
this.offlineModeToggle();
|
||||||
break;
|
break;
|
||||||
@@ -1316,13 +1493,14 @@ class Reader {
|
|||||||
if (!result && event.type == 'keydown') {
|
if (!result && event.type == 'keydown') {
|
||||||
const action = this.$root.readerActionByKeyEvent(event);
|
const action = this.$root.readerActionByKeyEvent(event);
|
||||||
|
|
||||||
if (action == 'loader') {
|
/*if (action == 'loader') {
|
||||||
result = this.doAction({action, event});
|
result = this.doAction({action, event});
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!result && this.activePage == 'TextPage') {
|
if (!result && this.activePage == 'TextPage') {
|
||||||
result = this.doAction({action, event});
|
result = this.doAction({action, event});
|
||||||
}
|
}*/
|
||||||
|
result = this.doAction({action, event});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return result;
|
return result;
|
||||||
@@ -1335,12 +1513,33 @@ export default vueComponent(Reader);
|
|||||||
|
|
||||||
<style scoped>
|
<style scoped>
|
||||||
.header {
|
.header {
|
||||||
|
height: 50px;
|
||||||
padding-left: 5px;
|
padding-left: 5px;
|
||||||
padding-right: 5px;
|
padding-right: 5px;
|
||||||
background-color: #1B695F;
|
background-color: #1B695F;
|
||||||
color: #000;
|
color: #000;
|
||||||
overflow: hidden;
|
overflow-x: auto;
|
||||||
height: 50px;
|
overflow-y: hidden;
|
||||||
|
scrollbar-color: #c49a60 #e4e4e4;
|
||||||
|
}
|
||||||
|
|
||||||
|
.header::-webkit-scrollbar {
|
||||||
|
height: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.header::-webkit-scrollbar-track {
|
||||||
|
background-color: #e4e4e4;
|
||||||
|
border-radius: 4px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.header::-webkit-scrollbar-thumb {
|
||||||
|
background-color: #c49a60;
|
||||||
|
border-radius: 4px;
|
||||||
|
border: 2px solid #e4e4e4;
|
||||||
|
}
|
||||||
|
|
||||||
|
.header::-webkit-scrollbar-thumb:hover {
|
||||||
|
background-color: #b48a50;
|
||||||
}
|
}
|
||||||
|
|
||||||
.main {
|
.main {
|
||||||
|
|||||||
@@ -5,12 +5,17 @@
|
|||||||
Что нового:
|
Что нового:
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<div style="line-height: 20px" v-html="whatsNewContent"></div>
|
<div style="line-height: 20px; min-width: 300px">
|
||||||
|
<div v-html="whatsNewContent"></div>
|
||||||
|
</div>
|
||||||
|
|
||||||
<span class="clickable" @click="openVersionHistory">Посмотреть историю версий</span>
|
<span class="clickable" style="font-size: 13px" @click="openVersionHistory">Посмотреть историю версий</span>
|
||||||
<span slot="footer">
|
|
||||||
<q-btn class="q-px-md" dense no-caps @click="whatsNewDisable">Больше не показывать</q-btn>
|
<template #footer>
|
||||||
</span>
|
<q-btn class="q-px-md" dense no-caps @click="whatsNewDisable">
|
||||||
|
Больше не показывать
|
||||||
|
</q-btn>
|
||||||
|
</template>
|
||||||
</Dialog>
|
</Dialog>
|
||||||
|
|
||||||
<Dialog ref="dialog2" v-model="donationVisible">
|
<Dialog ref="dialog2" v-model="donationVisible">
|
||||||
@@ -49,17 +54,40 @@
|
|||||||
|
|
||||||
<br><br>
|
<br><br>
|
||||||
<div class="row justify-center">
|
<div class="row justify-center">
|
||||||
<q-btn class="q-px-sm" color="primary" dense no-caps rounded @click="openDonate">
|
<!--q-btn class="q-px-sm" color="primary" dense no-caps @click="openDonate">
|
||||||
Помочь проекту
|
Помочь проекту
|
||||||
</q-btn>
|
</q-btn-->
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<span slot="footer">
|
<template #footer>
|
||||||
<span class="clickable row justify-end" style="font-size: 60%; color: grey" @click="donationDialogDisable">Больше не показывать</span>
|
<span class="clickable row justify-end" style="font-size: 60%; color: grey" @click="donationDialogDisable">Больше не показывать</span>
|
||||||
<br>
|
<br>
|
||||||
<q-btn class="q-px-sm" dense no-caps @click="donationDialogRemind">Напомнить позже</q-btn>
|
<q-btn class="q-px-sm" dense no-caps @click="donationDialogRemind">
|
||||||
</span>
|
Напомнить позже
|
||||||
|
</q-btn>
|
||||||
|
</template>
|
||||||
|
</Dialog>
|
||||||
|
|
||||||
|
<Dialog ref="dialog3" v-model="urlHelpVisible">
|
||||||
|
<template #header>
|
||||||
|
Обнаружена невалидная ссылка в поле "URL книги".
|
||||||
|
<br>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<div style="word-break: normal">
|
||||||
|
Если вы хотите найти определенную книгу и открыть в читалке, добро пожаловать в
|
||||||
|
раздел "Сетевая библиотека" (кнопка <q-icon name="la la-sitemap" size="32px" />) на сайте
|
||||||
|
<a href="https://liberama.top" target="_blank">liberama.top</a>
|
||||||
|
|
||||||
|
<br><br>
|
||||||
|
Если же вы пытаетесь вставить текст в читалку из буфера обмена, пожалуйста воспользуйтесь кнопкой
|
||||||
|
<q-btn no-caps dense class="q-px-sm" color="primary" size="13px" @click="loadBufferClick">
|
||||||
|
<q-icon class="q-mr-xs" name="la la-comment" size="24px" />
|
||||||
|
Из буфера обмена
|
||||||
|
</q-btn>
|
||||||
|
на странице загрузки.
|
||||||
|
</div>
|
||||||
</Dialog>
|
</Dialog>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
@@ -88,6 +116,7 @@ class ReaderDialogs {
|
|||||||
whatsNewVisible = false;
|
whatsNewVisible = false;
|
||||||
whatsNewContent = '';
|
whatsNewContent = '';
|
||||||
donationVisible = false;
|
donationVisible = false;
|
||||||
|
urlHelpVisible = false;
|
||||||
|
|
||||||
created() {
|
created() {
|
||||||
this.commit = this.$store.commit;
|
this.commit = this.$store.commit;
|
||||||
@@ -112,9 +141,9 @@ class ReaderDialogs {
|
|||||||
const whatsNew = versionHistory[0];
|
const whatsNew = versionHistory[0];
|
||||||
if (this.showWhatsNewDialog &&
|
if (this.showWhatsNewDialog &&
|
||||||
whatsNew.showUntil >= utils.formatDate(new Date(), 'coDate') &&
|
whatsNew.showUntil >= utils.formatDate(new Date(), 'coDate') &&
|
||||||
whatsNew.header != this.whatsNewContentHash) {
|
this.whatsNewHeader != this.whatsNewContentHash) {
|
||||||
await utils.sleep(2000);
|
await utils.sleep(2000);
|
||||||
this.whatsNewContent = 'Версия ' + whatsNew.header + whatsNew.content;
|
this.whatsNewContent = 'Версия ' + this.whatsNewHeader + whatsNew.content;
|
||||||
this.whatsNewVisible = true;
|
this.whatsNewVisible = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -128,6 +157,15 @@ class ReaderDialogs {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async showUrlHelp() {
|
||||||
|
this.urlHelpVisible = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
loadBufferClick() {
|
||||||
|
this.$emit('load-buffer-toggle');
|
||||||
|
this.urlHelpVisible = false;
|
||||||
|
}
|
||||||
|
|
||||||
donationDialogDisable() {
|
donationDialogDisable() {
|
||||||
this.donationVisible = false;
|
this.donationVisible = false;
|
||||||
if (this.showDonationDialog2020) {
|
if (this.showDonationDialog2020) {
|
||||||
@@ -160,8 +198,11 @@ class ReaderDialogs {
|
|||||||
|
|
||||||
whatsNewDisable() {
|
whatsNewDisable() {
|
||||||
this.whatsNewVisible = false;
|
this.whatsNewVisible = false;
|
||||||
const whatsNew = versionHistory[0];
|
this.commit('reader/setWhatsNewContentHash', this.whatsNewHeader);
|
||||||
this.commit('reader/setWhatsNewContentHash', whatsNew.header);
|
}
|
||||||
|
|
||||||
|
get whatsNewHeader() {
|
||||||
|
return `${versionHistory[0].version} (${versionHistory[0].releaseDate})`;
|
||||||
}
|
}
|
||||||
|
|
||||||
get mode() {
|
get mode() {
|
||||||
@@ -181,7 +222,7 @@ class ReaderDialogs {
|
|||||||
}
|
}
|
||||||
|
|
||||||
keyHook() {
|
keyHook() {
|
||||||
if (this.$refs.dialog1.active || this.$refs.dialog2.active)
|
if (this.$refs.dialog1.active || this.$refs.dialog2.active || this.$refs.dialog3.active)
|
||||||
return true;
|
return true;
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -8,12 +8,10 @@
|
|||||||
<span v-show="initStep">{{ initPercentage }}%</span>
|
<span v-show="initStep">{{ initPercentage }}%</span>
|
||||||
|
|
||||||
<div v-show="!initStep" class="input">
|
<div v-show="!initStep" class="input">
|
||||||
<!--input ref="input"
|
<q-input
|
||||||
placeholder="что ищем"
|
ref="input" v-model="needle"
|
||||||
:value="needle" @input="needle = $event.target.value"/-->
|
|
||||||
<q-input ref="input" v-model="needle"
|
|
||||||
class="col" outlined dense
|
class="col" outlined dense
|
||||||
placeholder="что ищем"
|
placeholder="Найти"
|
||||||
@keydown="inputKeyDown"
|
@keydown="inputKeyDown"
|
||||||
/>
|
/>
|
||||||
<div style="position: absolute; right: 10px; margin-top: 10px; font-size: 16px;">
|
<div style="position: absolute; right: 10px; margin-top: 10px; font-size: 16px;">
|
||||||
@@ -108,7 +106,7 @@ class SearchPage {
|
|||||||
this.parsed = parsed;
|
this.parsed = parsed;
|
||||||
}
|
}
|
||||||
|
|
||||||
this.header = 'Найти';
|
this.header = 'Поиск в тексте';
|
||||||
await this.$nextTick();
|
await this.$nextTick();
|
||||||
this.$refs.input.focus();
|
this.$refs.input.focus();
|
||||||
this.$refs.input.select();
|
this.$refs.input.select();
|
||||||
|
|||||||
@@ -576,7 +576,7 @@ class ServerStorage {
|
|||||||
newRecentPatch.rev++;
|
newRecentPatch.rev++;
|
||||||
newRecentPatch.data[itemKey] = _.cloneDeep(bm.recent[itemKey]);
|
newRecentPatch.data[itemKey] = _.cloneDeep(bm.recent[itemKey]);
|
||||||
|
|
||||||
let 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])
|
||||||
newRecentPatch.data[applyMod.key] = utils.applyObjDiff(newRecentPatch.data[applyMod.key], applyMod.mod, {isAddChanged: true});
|
newRecentPatch.data[applyMod.key] = utils.applyObjDiff(newRecentPatch.data[applyMod.key], applyMod.mod, {isAddChanged: true});
|
||||||
|
|
||||||
@@ -627,7 +627,7 @@ class ServerStorage {
|
|||||||
this.warning(`Последние изменения отменены. Данные синхронизированы с сервером.`);
|
this.warning(`Последние изменения отменены. Данные синхронизированы с сервером.`);
|
||||||
if (!recurse && itemKey) {
|
if (!recurse && itemKey) {
|
||||||
this.savingRecent = false;
|
this.savingRecent = false;
|
||||||
this.saveRecent(itemKey, true);
|
await this.saveRecent(itemKey, true);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
} else if (result.state == 'success') {
|
} else if (result.state == 'success') {
|
||||||
@@ -728,10 +728,10 @@ class ServerStorage {
|
|||||||
const ids = id.split('.');
|
const ids = id.split('.');
|
||||||
if (!(ids.length == 2) || !(ids[0] == this.hashedStorageKey))
|
if (!(ids.length == 2) || !(ids[0] == this.hashedStorageKey))
|
||||||
throw new Error(`decodeStorageItems: bad id - ${id}`);
|
throw new Error(`decodeStorageItems: bad id - ${id}`);
|
||||||
items[utils.fromBase58(ids[1])] = decoded;
|
items[utils.fromBase58(ids[1]).toString()] = decoded;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
result.items = items;
|
result.items = items;
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,9 +0,0 @@
|
|||||||
<div class="part-header">Показывать кнопки панели</div>
|
|
||||||
|
|
||||||
<div class="item row" v-for="item in toolButtons" :key="item.name" v-show="item.name != 'libs' || mode == 'liberama.top'">
|
|
||||||
<div class="label-3"></div>
|
|
||||||
<div class="col row">
|
|
||||||
<q-checkbox size="xs" v-model="showToolButton[item.name]" :label="rstore.readerActions[item.name]"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
@@ -52,7 +52,7 @@
|
|||||||
</q-checkbox>
|
</q-checkbox>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="item row">
|
<!--div class="item row">
|
||||||
<div class="label-6">Уведомление</div>
|
<div class="label-6">Уведомление</div>
|
||||||
<q-checkbox size="xs" v-model="showDonationDialog2020">
|
<q-checkbox size="xs" v-model="showDonationDialog2020">
|
||||||
Показывать "Оплатим хостинг вместе"
|
Показывать "Оплатим хостинг вместе"
|
||||||
@@ -60,7 +60,7 @@
|
|||||||
Показывать уведомление "Оплатим хостинг вместе"
|
Показывать уведомление "Оплатим хостинг вместе"
|
||||||
</q-tooltip>
|
</q-tooltip>
|
||||||
</q-checkbox>
|
</q-checkbox>
|
||||||
</div>
|
</div-->
|
||||||
|
|
||||||
<!---------------------------------------------->
|
<!---------------------------------------------->
|
||||||
<div class="part-header">Другое</div>
|
<div class="part-header">Другое</div>
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
<template>
|
<template>
|
||||||
<Window ref="window" height="95%" width="600px" @close="close">
|
<Window ref="window" width="600px" @close="close">
|
||||||
<template #header>
|
<template #header>
|
||||||
Настройки
|
Настройки
|
||||||
</template>
|
</template>
|
||||||
@@ -24,7 +24,7 @@
|
|||||||
<div v-show="tabsScrollable" class="q-pt-lg" />
|
<div v-show="tabsScrollable" class="q-pt-lg" />
|
||||||
<q-tab class="tab" name="profiles" icon="la la-users" label="Профили" />
|
<q-tab class="tab" name="profiles" icon="la la-users" label="Профили" />
|
||||||
<q-tab class="tab" name="view" icon="la la-eye" label="Вид" />
|
<q-tab class="tab" name="view" icon="la la-eye" label="Вид" />
|
||||||
<q-tab class="tab" name="buttons" icon="la la-grip-horizontal" label="Кнопки" />
|
<q-tab class="tab" name="toolbar" icon="la la-grip-horizontal" label="Панель" />
|
||||||
<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="Конвертир." />
|
||||||
@@ -82,8 +82,8 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<!-- Кнопки ---------------------------------------------------------------------->
|
<!-- Кнопки ---------------------------------------------------------------------->
|
||||||
<div v-if="selectedTab == 'buttons'" class="fit tab-panel">
|
<div v-if="selectedTab == 'toolbar'" class="fit tab-panel">
|
||||||
@@include('./ButtonsTab.inc');
|
@@include('./ToolBarTab.inc');
|
||||||
</div>
|
</div>
|
||||||
<!-- Управление ------------------------------------------------------------------>
|
<!-- Управление ------------------------------------------------------------------>
|
||||||
<div v-if="selectedTab == 'keys'" class="fit column">
|
<div v-if="selectedTab == 'keys'" class="fit column">
|
||||||
@@ -124,6 +124,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';
|
||||||
|
|
||||||
@@ -636,8 +637,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;
|
||||||
@@ -702,11 +712,11 @@ export default vueComponent(SettingsPage);
|
|||||||
margin-bottom: 5px;
|
margin-bottom: 5px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.label-1, .label-7 {
|
.label-1, .label-3, .label-7 {
|
||||||
width: 75px;
|
width: 75px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.label-2, .label-3, .label-4, .label-5 {
|
.label-2, .label-4, .label-5 {
|
||||||
width: 110px;
|
width: 110px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
18
client/components/Reader/SettingsPage/ToolBarTab.inc
Normal file
18
client/components/Reader/SettingsPage/ToolBarTab.inc
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
<div class="part-header">Отображение</div>
|
||||||
|
|
||||||
|
<div class="item row no-wrap">
|
||||||
|
<div class="label-3"></div>
|
||||||
|
<q-checkbox size="xs" v-model="toolBarHideOnScroll" label="Скрывать/показывать панель при прокрутке" >
|
||||||
|
<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="item row no-wrap" v-for="item in toolButtons" :key="item.name" v-show="item.name != 'libs' || mode == 'liberama.top'">
|
||||||
|
<div class="label-3"></div>
|
||||||
|
<q-checkbox size="xs" v-model="showToolButton[item.name]" :label="rstore.readerActions[item.name]"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
@@ -13,7 +13,7 @@
|
|||||||
ref="input"
|
ref="input"
|
||||||
v-model="search"
|
v-model="search"
|
||||||
class="q-ml-sm col"
|
class="q-ml-sm col"
|
||||||
outlined dense rounded
|
outlined dense
|
||||||
bg-color="grey-4"
|
bg-color="grey-4"
|
||||||
placeholder="Найти"
|
placeholder="Найти"
|
||||||
@click.stop
|
@click.stop
|
||||||
|
|||||||
@@ -6,29 +6,32 @@
|
|||||||
</div>
|
</div>
|
||||||
<div ref="scrollBox1" class="layout over-hidden" @wheel.prevent.stop="onMouseWheel">
|
<div ref="scrollBox1" class="layout over-hidden" @wheel.prevent.stop="onMouseWheel">
|
||||||
<div ref="scrollingPage1" class="layout over-hidden" @transitionend="onPage1TransitionEnd" @animationend="onPage1AnimationEnd">
|
<div ref="scrollingPage1" class="layout over-hidden" @transitionend="onPage1TransitionEnd" @animationend="onPage1AnimationEnd">
|
||||||
<div v-html="page1"></div>
|
<div @copy.prevent="copyText" v-html="page1"></div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div ref="scrollBox2" class="layout over-hidden" @wheel.prevent.stop="onMouseWheel">
|
<div ref="scrollBox2" class="layout over-hidden" @wheel.prevent.stop="onMouseWheel">
|
||||||
<div ref="scrollingPage2" class="layout over-hidden" @transitionend="onPage2TransitionEnd" @animationend="onPage2AnimationEnd">
|
<div ref="scrollingPage2" class="layout over-hidden" @transitionend="onPage2TransitionEnd" @animationend="onPage2AnimationEnd">
|
||||||
<div v-html="page2"></div>
|
<div @copy.prevent="copyText" v-html="page2"></div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div v-show="showStatusBar" ref="statusBar" class="layout">
|
<div v-show="showStatusBar" ref="statusBar" class="layout">
|
||||||
<div v-html="statusBar"></div>
|
<div v-html="statusBar"></div>
|
||||||
</div>
|
</div>
|
||||||
<div v-show="clickControl" ref="layoutEvents" class="layout events"
|
<div
|
||||||
|
v-show="clickControl" ref="layoutEvents" class="layout events"
|
||||||
oncontextmenu="return false;"
|
oncontextmenu="return false;"
|
||||||
@mousedown.prevent.stop="onMouseDown" @mouseup.prevent.stop="onMouseUp"
|
@mousedown.prevent.stop="onMouseDown" @mouseup.prevent.stop="onMouseUp"
|
||||||
@wheel.prevent.stop="onMouseWheel"
|
@wheel.prevent.stop="onMouseWheel"
|
||||||
@touchstart.stop="onTouchStart" @touchend.stop="onTouchEnd" @touchmove.stop="onTouchMove" @touchcancel.prevent.stop="onTouchCancel"
|
@touchstart.stop="onTouchStart" @touchend.stop="onTouchEnd" @touchmove.stop="onTouchMove" @touchcancel.prevent.stop="onTouchCancel"
|
||||||
>
|
>
|
||||||
<div v-show="showStatusBar && statusBarClickOpen" @mousedown.prevent.stop @touchstart.stop
|
<div
|
||||||
|
v-show="showStatusBar && statusBarClickOpen" @mousedown.prevent.stop @touchstart.stop
|
||||||
@click.prevent.stop="onStatusBarClick"
|
@click.prevent.stop="onStatusBarClick"
|
||||||
v-html="statusBarClickable"
|
v-html="statusBarClickable"
|
||||||
></div>
|
></div>
|
||||||
</div>
|
</div>
|
||||||
<div v-show="!clickControl && showStatusBar && statusBarClickOpen" class="layout"
|
<div
|
||||||
|
v-show="!clickControl && showStatusBar && statusBarClickOpen" class="layout"
|
||||||
@mousedown.prevent.stop @touchstart.stop
|
@mousedown.prevent.stop @touchstart.stop
|
||||||
@click.prevent.stop="onStatusBarClick"
|
@click.prevent.stop="onStatusBarClick"
|
||||||
v-html="statusBarClickable"
|
v-html="statusBarClickable"
|
||||||
@@ -46,6 +49,7 @@ import vueComponent from '../../vueComponent.js';
|
|||||||
|
|
||||||
import {loadCSS} from 'fg-loadcss';
|
import {loadCSS} from 'fg-loadcss';
|
||||||
import _ from 'lodash';
|
import _ from 'lodash';
|
||||||
|
import he from 'he';
|
||||||
|
|
||||||
import './TextPage.css';
|
import './TextPage.css';
|
||||||
|
|
||||||
@@ -62,7 +66,14 @@ const componentOptions = {
|
|||||||
watch: {
|
watch: {
|
||||||
bookPos: function() {
|
bookPos: function() {
|
||||||
this.$emit('book-pos-changed', {bookPos: this.bookPos, bookPosSeen: this.bookPosSeen});
|
this.$emit('book-pos-changed', {bookPos: this.bookPos, bookPosSeen: this.bookPosSeen});
|
||||||
|
|
||||||
this.draw();
|
this.draw();
|
||||||
|
|
||||||
|
if (this.userBookPosChange) {
|
||||||
|
this.$emit('hide-tool-bar', {show: (this.bookPos == 0 || this.bookPos < this.prevBookPos)});
|
||||||
|
this.prevBookPos = this.bookPos;
|
||||||
|
this.userBookPosChange = false;
|
||||||
|
}
|
||||||
},
|
},
|
||||||
bookPosSeen: function() {
|
bookPosSeen: function() {
|
||||||
this.$emit('book-pos-changed', {bookPos: this.bookPos, bookPosSeen: this.bookPosSeen});
|
this.$emit('book-pos-changed', {bookPos: this.bookPos, bookPosSeen: this.bookPosSeen});
|
||||||
@@ -95,6 +106,8 @@ class TextPage {
|
|||||||
lastBook = null;
|
lastBook = null;
|
||||||
bookPos = 0;
|
bookPos = 0;
|
||||||
bookPosSeen = null;
|
bookPosSeen = null;
|
||||||
|
prevBookPos = 0;
|
||||||
|
userBookPosChange = false;
|
||||||
|
|
||||||
fontStyle = null;
|
fontStyle = null;
|
||||||
fontSize = null;
|
fontSize = null;
|
||||||
@@ -151,7 +164,7 @@ class TextPage {
|
|||||||
|
|
||||||
this.$root.addEventHook('resize', async() => {
|
this.$root.addEventHook('resize', async() => {
|
||||||
this.$nextTick(this.onResize);
|
this.$nextTick(this.onResize);
|
||||||
await utils.sleep(500);
|
await utils.sleep(200);
|
||||||
this.$nextTick(this.onResize);
|
this.$nextTick(this.onResize);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@@ -495,12 +508,25 @@ class TextPage {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async onResize() {
|
async onResize() {
|
||||||
|
if (this.resizing)
|
||||||
|
return;
|
||||||
|
|
||||||
|
this.resizing = true;
|
||||||
try {
|
try {
|
||||||
|
const scrolled = this.doingScrolling;
|
||||||
|
if (scrolled)
|
||||||
|
await this.stopTextScrolling();
|
||||||
|
|
||||||
this.calcDrawProps();
|
this.calcDrawProps();
|
||||||
this.setBackground();
|
this.setBackground();
|
||||||
this.draw();
|
this.draw();
|
||||||
|
|
||||||
|
if (scrolled)
|
||||||
|
this.startTextScrolling();
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
//
|
//
|
||||||
|
} finally {
|
||||||
|
this.resizing = false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -648,7 +674,7 @@ class TextPage {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (this.book && this.bookPos > 0 && this.bookPos >= this.parsed.textLength) {
|
if (this.book && this.bookPos > 0 && this.bookPos >= this.parsed.textLength) {
|
||||||
this.doEnd(true);
|
this.doEnd(true, false);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -671,7 +697,7 @@ class TextPage {
|
|||||||
this.debouncedDrawPageDividerAndOrnament();
|
this.debouncedDrawPageDividerAndOrnament();
|
||||||
|
|
||||||
if (this.book && this.linesDown && this.linesDown.length < this.pageLineCount) {
|
if (this.book && this.linesDown && this.linesDown.length < this.pageLineCount) {
|
||||||
this.doEnd(true);
|
this.doEnd(true, false);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -907,12 +933,14 @@ class TextPage {
|
|||||||
|
|
||||||
doDown() {
|
doDown() {
|
||||||
if (this.linesDown && this.linesDown.length > this.pageLineCount && this.pageLineCount > 0) {
|
if (this.linesDown && this.linesDown.length > this.pageLineCount && this.pageLineCount > 0) {
|
||||||
|
this.userBookPosChange = true;
|
||||||
this.bookPos = this.linesDown[1].begin;
|
this.bookPos = this.linesDown[1].begin;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
doUp() {
|
doUp() {
|
||||||
if (this.linesUp && this.linesUp.length > 1 && this.pageLineCount > 0) {
|
if (this.linesUp && this.linesUp.length > 1 && this.pageLineCount > 0) {
|
||||||
|
this.userBookPosChange = true;
|
||||||
this.bookPos = this.linesUp[1].begin;
|
this.bookPos = this.linesUp[1].begin;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -925,6 +953,7 @@ class TextPage {
|
|||||||
if (i >= 0 && this.linesDown.length >= 2*i + (this.keepLastToFirst ? 1 : 0)) {
|
if (i >= 0 && this.linesDown.length >= 2*i + (this.keepLastToFirst ? 1 : 0)) {
|
||||||
this.currentAnimation = this.pageChangeAnimation;
|
this.currentAnimation = this.pageChangeAnimation;
|
||||||
this.pageChangeDirectionDown = true;
|
this.pageChangeDirectionDown = true;
|
||||||
|
this.userBookPosChange = true;
|
||||||
this.bookPos = this.linesDown[i].begin;
|
this.bookPos = this.linesDown[i].begin;
|
||||||
} else
|
} else
|
||||||
this.doEnd();
|
this.doEnd();
|
||||||
@@ -940,6 +969,7 @@ class TextPage {
|
|||||||
if (i >= 0 && this.linesUp.length > i) {
|
if (i >= 0 && this.linesUp.length > i) {
|
||||||
this.currentAnimation = this.pageChangeAnimation;
|
this.currentAnimation = this.pageChangeAnimation;
|
||||||
this.pageChangeDirectionDown = false;
|
this.pageChangeDirectionDown = false;
|
||||||
|
this.userBookPosChange = true;
|
||||||
this.bookPos = this.linesUp[i].begin;
|
this.bookPos = this.linesUp[i].begin;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -948,10 +978,11 @@ class TextPage {
|
|||||||
doHome() {
|
doHome() {
|
||||||
this.currentAnimation = this.pageChangeAnimation;
|
this.currentAnimation = this.pageChangeAnimation;
|
||||||
this.pageChangeDirectionDown = false;
|
this.pageChangeDirectionDown = false;
|
||||||
|
this.userBookPosChange = true;
|
||||||
this.bookPos = 0;
|
this.bookPos = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
doEnd(noAni) {
|
doEnd(noAni, isUser = true) {
|
||||||
if (this.parsed.para.length && this.pageLineCount > 0) {
|
if (this.parsed.para.length && this.pageLineCount > 0) {
|
||||||
let i = this.parsed.para.length - 1;
|
let i = this.parsed.para.length - 1;
|
||||||
let lastPos = this.parsed.para[i].offset + this.parsed.para[i].length - 1;
|
let lastPos = this.parsed.para[i].offset + this.parsed.para[i].length - 1;
|
||||||
@@ -962,6 +993,7 @@ class TextPage {
|
|||||||
if (!noAni)
|
if (!noAni)
|
||||||
this.currentAnimation = this.pageChangeAnimation;
|
this.currentAnimation = this.pageChangeAnimation;
|
||||||
this.pageChangeDirectionDown = true;
|
this.pageChangeDirectionDown = true;
|
||||||
|
this.userBookPosChange = isUser;
|
||||||
this.bookPos = lines[i].begin;
|
this.bookPos = lines[i].begin;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -1201,8 +1233,54 @@ class TextPage {
|
|||||||
}
|
}
|
||||||
|
|
||||||
return action;
|
return action;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
copyText(event) {
|
||||||
|
//все это для того, чтобы правильно расставить переносы \n при копировании текста
|
||||||
|
//прямо с текущей страницы
|
||||||
|
|
||||||
|
//подготовка, вытаскиваем весь текст страницы
|
||||||
|
const lines = this.getLines(this.bookPos);
|
||||||
|
const decodedLines = [];
|
||||||
|
for (const line of lines.linesDown) {
|
||||||
|
let lineText = '';
|
||||||
|
for (const part of line.parts) {
|
||||||
|
lineText += part.text;
|
||||||
|
}
|
||||||
|
decodedLines.push({text: he.decode(lineText), first: line.first});
|
||||||
|
}
|
||||||
|
|
||||||
|
let i = 0;
|
||||||
|
const findDecoded = (line) => {
|
||||||
|
for (let j = i; j < decodedLines.length; j++) {
|
||||||
|
const decoded = decodedLines[j];
|
||||||
|
if (decoded.text.indexOf(line) >= 0) {
|
||||||
|
i = j;
|
||||||
|
return decoded;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const selection = document.getSelection();
|
||||||
|
const splitted = selection.toString().split(/[\n\r]/);
|
||||||
|
|
||||||
|
let filtered = '';
|
||||||
|
//формируем filtered, учитывая переносы из decodedLines
|
||||||
|
for (const line of splitted) {
|
||||||
|
const found = findDecoded(line);
|
||||||
|
if (found && found.first) {
|
||||||
|
filtered += (filtered ? '\n' : '') + line;
|
||||||
|
} else {
|
||||||
|
filtered += (filtered ? '\r ' : '') + line;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//маленькие хитрости, убираем переносы по слогам
|
||||||
|
filtered = filtered.replace(/-\r /g, '').replace(/\r /g, ' ');
|
||||||
|
|
||||||
|
event.clipboardData.setData('text/plain', filtered);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export default vueComponent(TextPage);
|
export default vueComponent(TextPage);
|
||||||
|
|||||||
@@ -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);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,10 +1,14 @@
|
|||||||
import localForage from 'localforage';
|
import localForage from 'localforage';
|
||||||
|
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;
|
||||||
|
|
||||||
//локальный кэш метаданных книг, ограничение maxDataSize
|
//локальный кэш метаданных книг, ограничение maxDataSize
|
||||||
const bmMetaStore = localForage.createInstance({
|
const bmMetaStore = localForage.createInstance({
|
||||||
@@ -17,9 +21,6 @@ const bmDataStore = localForage.createInstance({
|
|||||||
});
|
});
|
||||||
|
|
||||||
//список недавно открытых книг
|
//список недавно открытых книг
|
||||||
const bmRecentStoreOld = localForage.createInstance({
|
|
||||||
name: 'bmRecentStore'
|
|
||||||
});
|
|
||||||
const bmRecentStoreNew = localForage.createInstance({
|
const bmRecentStoreNew = localForage.createInstance({
|
||||||
name: 'bmRecentStoreNew'
|
name: 'bmRecentStoreNew'
|
||||||
});
|
});
|
||||||
@@ -39,7 +40,7 @@ class BookManager {
|
|||||||
|
|
||||||
this.saveRecentItem = _.debounce(() => {
|
this.saveRecentItem = _.debounce(() => {
|
||||||
bmRecentStoreNew.setItem('recent-item', this.recentItem);
|
bmRecentStoreNew.setItem('recent-item', this.recentItem);
|
||||||
this.recentRev = (this.recentRev < 1000 ? this.recentRev + 1 : 1);
|
this.recentRev = (this.recentRev < maxRecentLength ? this.recentRev + 1 : 1);
|
||||||
bmRecentStoreNew.setItem('rev', this.recentRev);
|
bmRecentStoreNew.setItem('rev', this.recentRev);
|
||||||
}, 200, {maxWait: 300});
|
}, 200, {maxWait: 300});
|
||||||
|
|
||||||
@@ -54,6 +55,9 @@ class BookManager {
|
|||||||
if (this.recentItem)
|
if (this.recentItem)
|
||||||
this.recent[this.recentItem.key] = this.recentItem;
|
this.recent[this.recentItem.key] = this.recentItem;
|
||||||
|
|
||||||
|
//конвертируем в новые ключи
|
||||||
|
await this.convertRecent();
|
||||||
|
|
||||||
this.recentLastKey = await bmRecentStoreNew.getItem('recent-last-key');
|
this.recentLastKey = await bmRecentStoreNew.getItem('recent-last-key');
|
||||||
if (this.recentLastKey) {
|
if (this.recentLastKey) {
|
||||||
const meta = await bmMetaStore.getItem(`bmMeta-${this.recentLastKey}`);
|
const meta = await bmMetaStore.getItem(`bmMeta-${this.recentLastKey}`);
|
||||||
@@ -63,48 +67,6 @@ class BookManager {
|
|||||||
}
|
}
|
||||||
|
|
||||||
await this.cleanRecentBooks();
|
await this.cleanRecentBooks();
|
||||||
|
|
||||||
//TODO: убрать после 06.2021, когда bmRecentStoreOld устареет
|
|
||||||
{
|
|
||||||
await this.convertFileToDiskPrefix();
|
|
||||||
if (this.recentRev > 10)
|
|
||||||
await bmRecentStoreOld.clear();
|
|
||||||
}
|
|
||||||
} else {//TODO: убрать после 06.2021, когда bmRecentStoreOld устареет
|
|
||||||
this.recentLast = await bmRecentStoreOld.getItem('recent-last');
|
|
||||||
if (this.recentLast) {
|
|
||||||
this.recent[this.recentLast.key] = this.recentLast;
|
|
||||||
const meta = await bmMetaStore.getItem(`bmMeta-${this.recentLast.key}`);
|
|
||||||
if (_.isObject(meta)) {
|
|
||||||
this.books[meta.key] = meta;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
let key = null;
|
|
||||||
const len = await bmRecentStoreOld.length();
|
|
||||||
for (let i = len - 1; i >= 0; i--) {
|
|
||||||
key = await bmRecentStoreOld.key(i);
|
|
||||||
if (key) {
|
|
||||||
let r = await bmRecentStoreOld.getItem(key);
|
|
||||||
if (_.isObject(r) && r.key) {
|
|
||||||
this.recent[r.key] = r;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
await bmRecentStoreOld.removeItem(key);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
//размножение для дебага
|
|
||||||
/*if (key) {
|
|
||||||
for (let i = 0; i < 1000; i++) {
|
|
||||||
const k = this.keyFromUrl(i.toString());
|
|
||||||
this.recent[k] = Object.assign({}, _.cloneDeep(this.recent[key]), {key: k, touchTime: Date.now() - 1000000, url: utils.randomHexString(300)});
|
|
||||||
}
|
|
||||||
}*/
|
|
||||||
|
|
||||||
await bmRecentStoreNew.setItem('recent', this.recent);
|
|
||||||
this.recentRev = 1;
|
|
||||||
await bmRecentStoreNew.setItem('rev', this.recentRev);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
this.recentChanged = true;
|
this.recentChanged = true;
|
||||||
@@ -112,6 +74,40 @@ class BookManager {
|
|||||||
this.loadStored();//no await
|
this.loadStored();//no await
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//TODO: убрать в 2025г
|
||||||
|
async convertRecent() {
|
||||||
|
const converted = await bmRecentStoreNew.getItem('recent-converted');
|
||||||
|
|
||||||
|
if (converted)
|
||||||
|
return;
|
||||||
|
|
||||||
|
const newRecent = {};
|
||||||
|
for (const book of Object.values(this.recent)) {
|
||||||
|
|
||||||
|
if (!book.path) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
const newKey = this.keyFromPath(book.path);
|
||||||
|
|
||||||
|
newRecent[newKey] = _.cloneDeep(book);
|
||||||
|
newRecent[newKey].key = newKey;
|
||||||
|
if (!newRecent[newKey].loadTime)
|
||||||
|
newRecent[newKey].loadTime = newRecent[newKey].addTime;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.recent = newRecent;
|
||||||
|
|
||||||
|
//console.log(converted);
|
||||||
|
(async() => {
|
||||||
|
await utils.sleep(3000);
|
||||||
|
this.saveRecent();
|
||||||
|
this.emit('recent-changed');
|
||||||
|
this.emit('set-recent');
|
||||||
|
await bmRecentStoreNew.setItem('recent-converted', true);
|
||||||
|
})();
|
||||||
|
}
|
||||||
|
|
||||||
//Ленивая асинхронная загрузка bmMetaStore
|
//Ленивая асинхронная загрузка bmMetaStore
|
||||||
async loadStored() {
|
async loadStored() {
|
||||||
//даем время для загрузки последней читаемой книги, чтобы не блокировать приложение
|
//даем время для загрузки последней читаемой книги, чтобы не блокировать приложение
|
||||||
@@ -238,8 +234,8 @@ 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};
|
||||||
meta.key = this.keyFromUrl(meta.url);
|
meta.key = this.keyFromPath(meta.path);
|
||||||
meta.addTime = Date.now();
|
meta.addTime = Date.now();//время добавления в кеш
|
||||||
|
|
||||||
const cb = (perc) => {
|
const cb = (perc) => {
|
||||||
const p = Math.round(30*perc/100);
|
const p = Math.round(30*perc/100);
|
||||||
@@ -274,10 +270,10 @@ class BookManager {
|
|||||||
async hasBookParsed(meta) {
|
async hasBookParsed(meta) {
|
||||||
if (!this.books)
|
if (!this.books)
|
||||||
return false;
|
return false;
|
||||||
if (!meta.url)
|
if (!meta.path)
|
||||||
return false;
|
return false;
|
||||||
if (!meta.key)
|
if (!meta.key)
|
||||||
meta.key = this.keyFromUrl(meta.url);
|
meta.key = this.keyFromPath(meta.path);
|
||||||
|
|
||||||
let book = this.books[meta.key];
|
let book = this.books[meta.key];
|
||||||
|
|
||||||
@@ -292,8 +288,12 @@ class BookManager {
|
|||||||
|
|
||||||
async getBook(meta, callback) {
|
async getBook(meta, callback) {
|
||||||
let result = undefined;
|
let result = undefined;
|
||||||
|
|
||||||
|
if (!meta.path)
|
||||||
|
return;
|
||||||
|
|
||||||
if (!meta.key)
|
if (!meta.key)
|
||||||
meta.key = this.keyFromUrl(meta.url);
|
meta.key = this.keyFromPath(meta.path);
|
||||||
|
|
||||||
result = this.books[meta.key];
|
result = this.books[meta.key];
|
||||||
|
|
||||||
@@ -303,11 +303,6 @@ class BookManager {
|
|||||||
this.books[meta.key] = result;
|
this.books[meta.key] = result;
|
||||||
}
|
}
|
||||||
|
|
||||||
//Если файл на сервере изменился, считаем, что в кеше его нету
|
|
||||||
if (meta.path && result && meta.path != result.path) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (result && !result.parsed) {
|
if (result && !result.parsed) {
|
||||||
let data = await bmDataStore.getItem(`bmData-${meta.key}`);
|
let data = await bmDataStore.getItem(`bmData-${meta.key}`);
|
||||||
callback(5);
|
callback(5);
|
||||||
@@ -352,9 +347,36 @@ 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);
|
||||||
|
}
|
||||||
|
|
||||||
|
//отправим dataUrl на сервер в /upload
|
||||||
|
try {
|
||||||
|
await readerApi.uploadFileBuf(dataUrl, (url) => {
|
||||||
|
coverPageUrl = url;
|
||||||
|
});
|
||||||
|
} catch (e) {
|
||||||
|
console.error(e);
|
||||||
|
}
|
||||||
|
|
||||||
|
//сохраним в storage
|
||||||
|
if (coverPageUrl)
|
||||||
|
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
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -367,14 +389,24 @@ class BookManager {
|
|||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
keyFromUrl(url) {
|
/*keyFromUrl(url) {
|
||||||
return utils.stringToHex(url);
|
return utils.stringToHex(url);
|
||||||
|
}*/
|
||||||
|
|
||||||
|
keyFromPath(bookPath) {
|
||||||
|
return path.basename(bookPath);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
keysEqual(bookPath1, bookPath2) {
|
||||||
|
if (bookPath1 === undefined || bookPath2 === undefined)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
return (this.keyFromPath(bookPath1) === this.keyFromPath(bookPath2));
|
||||||
|
}
|
||||||
//-- recent --------------------------------------------------------------
|
//-- recent --------------------------------------------------------------
|
||||||
async recentSetItem(item = null, skipCheck = false) {
|
async recentSetItem(item = null, skipCheck = false) {
|
||||||
const rev = await bmRecentStoreNew.getItem('rev');
|
const rev = await bmRecentStoreNew.getItem('rev');
|
||||||
if (rev != this.recentRev && !skipCheck) {
|
if (rev != this.recentRev && !skipCheck) {//если изменение произошло в другой вкладке барузера
|
||||||
const newRecent = await bmRecentStoreNew.getItem('recent');
|
const newRecent = await bmRecentStoreNew.getItem('recent');
|
||||||
Object.assign(this.recent, newRecent);
|
Object.assign(this.recent, newRecent);
|
||||||
this.recentItem = await bmRecentStoreNew.getItem('recent-item');
|
this.recentItem = await bmRecentStoreNew.getItem('recent-item');
|
||||||
@@ -411,7 +443,10 @@ class BookManager {
|
|||||||
|
|
||||||
async setRecentBook(value) {
|
async setRecentBook(value) {
|
||||||
let result = this.metaOnly(value);
|
let result = this.metaOnly(value);
|
||||||
result.touchTime = Date.now();
|
result.touchTime = Date.now();//время последнего чтения
|
||||||
|
if (!result.loadTime)
|
||||||
|
result.loadTime = Date.now();//время загрузки файла
|
||||||
|
|
||||||
result.deleted = 0;
|
result.deleted = 0;
|
||||||
|
|
||||||
if (this.recent[result.key]) {
|
if (this.recent[result.key]) {
|
||||||
@@ -427,9 +462,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);
|
||||||
@@ -439,11 +474,18 @@ 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 cleanRecentBooks() {
|
async cleanRecentBooks() {
|
||||||
const sorted = this.getSortedRecent();
|
const sorted = this.getSortedRecent();
|
||||||
|
|
||||||
let isDel = false;
|
let isDel = false;
|
||||||
for (let i = 1000; i < sorted.length; i++) {
|
for (let i = maxRecentLength; i < sorted.length; i++) {
|
||||||
delete this.recent[sorted[i].key];
|
delete this.recent[sorted[i].key];
|
||||||
isDel = true;
|
isDel = true;
|
||||||
}
|
}
|
||||||
@@ -455,33 +497,6 @@ class BookManager {
|
|||||||
return isDel;
|
return isDel;
|
||||||
}
|
}
|
||||||
|
|
||||||
async convertFileToDiskPrefix() {
|
|
||||||
let isConverted = false;
|
|
||||||
|
|
||||||
const newRecent = {};
|
|
||||||
for (let key of Object.keys(this.recent)) {
|
|
||||||
let newKey = key;
|
|
||||||
let newUrl = this.recent[key].url;
|
|
||||||
|
|
||||||
if (newKey.indexOf('66696c65') == 0) {
|
|
||||||
newKey = newKey.replace(/^66696c65/, '6469736b');
|
|
||||||
if (newUrl)
|
|
||||||
newUrl = newUrl.replace(/^file/, 'disk');
|
|
||||||
isConverted = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
newRecent[newKey] = this.recent[key];
|
|
||||||
newRecent[newKey].key = newKey;
|
|
||||||
if (newUrl)
|
|
||||||
newRecent[newKey].url = newUrl;
|
|
||||||
}
|
|
||||||
if (isConverted) {
|
|
||||||
this.recent = newRecent;
|
|
||||||
await this.recentSetItem(null, true);
|
|
||||||
}
|
|
||||||
return isConverted;
|
|
||||||
}
|
|
||||||
|
|
||||||
mostRecentBook() {
|
mostRecentBook() {
|
||||||
if (this.recentLastKey) {
|
if (this.recentLastKey) {
|
||||||
return this.recent[this.recentLastKey];
|
return this.recent[this.recentLastKey];
|
||||||
@@ -490,7 +505,7 @@ class BookManager {
|
|||||||
|
|
||||||
let max = 0;
|
let max = 0;
|
||||||
let result = null;
|
let result = null;
|
||||||
for (let key in this.recent) {
|
for (const key in this.recent) {
|
||||||
const book = this.recent[key];
|
const book = this.recent[key];
|
||||||
if (!book.deleted && book.touchTime > max) {
|
if (!book.deleted && book.touchTime > max) {
|
||||||
max = book.touchTime;
|
max = book.touchTime;
|
||||||
@@ -521,6 +536,43 @@ class BookManager {
|
|||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
findRecentByUrlAndPath(url, bookPath) {
|
||||||
|
if (bookPath) {
|
||||||
|
const key = this.keyFromPath(bookPath);
|
||||||
|
const book = this.recent[key];
|
||||||
|
if (book && !book.deleted)
|
||||||
|
return book;
|
||||||
|
}
|
||||||
|
|
||||||
|
let max = 0;
|
||||||
|
let result = null;
|
||||||
|
|
||||||
|
for (const key in this.recent) {
|
||||||
|
const book = this.recent[key];
|
||||||
|
if (!book.deleted && book.url == url && book.loadTime > max) {
|
||||||
|
max = book.loadTime;
|
||||||
|
result = book;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
findRecentBySameBookKey(sameKey) {
|
||||||
|
let max = 0;
|
||||||
|
let result = null;
|
||||||
|
|
||||||
|
for (const key in this.recent) {
|
||||||
|
const book = this.recent[key];
|
||||||
|
if (!book.deleted && book.sameBookKey == sameKey && book.loadTime > max) {
|
||||||
|
max = book.loadTime;
|
||||||
|
result = book;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
async setRecent(value) {
|
async setRecent(value) {
|
||||||
const mergedRecent = _.cloneDeep(this.recent);
|
const mergedRecent = _.cloneDeep(this.recent);
|
||||||
|
|
||||||
|
|||||||
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,62 +1,137 @@
|
|||||||
export const versionHistory = [
|
export const versionHistory = [
|
||||||
{
|
{
|
||||||
|
version: '0.11.8',
|
||||||
|
releaseDate: '2022-07-14',
|
||||||
|
showUntil: '2022-07-13',
|
||||||
|
content:
|
||||||
|
`
|
||||||
|
<ul>
|
||||||
|
<li>добавлено отображение и синхронизация обложек в окне загруженных книг</li>
|
||||||
|
<li>добавлена синхронизация обоев</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
`
|
||||||
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
version: '0.11.7',
|
||||||
|
releaseDate: '2022-07-12',
|
||||||
|
showUntil: '2022-07-19',
|
||||||
|
content:
|
||||||
|
`
|
||||||
|
<ul>
|
||||||
|
<li>добавлено автосокрытие панели управления при листании, отключается в настройках</li>
|
||||||
|
<li>изменения в окне загруженных книг:</li>
|
||||||
|
<ul>
|
||||||
|
<li>добавлена группировка по версиям файла одной и той же книги</li>
|
||||||
|
<li>группировка происходит по имени загружаемого файла, либо по URL книги</li>
|
||||||
|
<li>добавлены различные методы сортировки списка загруженных книг</li>
|
||||||
|
<li>нумерация всегда осуществляется по времени загрузки</li>
|
||||||
|
</ul>
|
||||||
|
<li>незначительные общие изменения интерфейса, приведение к единому стилю</li>
|
||||||
|
<li>исправления багов</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
`
|
||||||
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
version: '0.11.6',
|
||||||
|
releaseDate: '2022-07-02',
|
||||||
|
showUntil: '2022-07-01',
|
||||||
|
content:
|
||||||
|
`
|
||||||
|
<ul>
|
||||||
|
<li>улучшено копирование текста прямо со страницы, для переводчиков</li>
|
||||||
|
<li>актуализация используемых пакетов</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
`
|
||||||
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
version: '0.11.5',
|
||||||
|
releaseDate: '2022-04-15',
|
||||||
|
showUntil: '2022-04-14',
|
||||||
|
content:
|
||||||
|
`
|
||||||
|
<ul>
|
||||||
|
<li>небольшие дополнения интерфейса</li>
|
||||||
|
<li>исправления багов</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
`
|
||||||
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
version: '0.11.1',
|
||||||
|
releaseDate: '2021-12-03',
|
||||||
showUntil: '2021-12-02',
|
showUntil: '2021-12-02',
|
||||||
header: '0.11.1 (2021-12-03)',
|
|
||||||
content:
|
content:
|
||||||
`
|
`
|
||||||
<ul>
|
<ul>
|
||||||
<li>переход на JembaDb вместо SQLite</li>
|
<li>переход на JembaDb вместо SQLite</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
`
|
`
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
|
version: '0.11.0',
|
||||||
|
releaseDate: '2021-11-18',
|
||||||
showUntil: '2021-11-17',
|
showUntil: '2021-11-17',
|
||||||
header: '0.11.0 (2021-11-18)',
|
|
||||||
content:
|
content:
|
||||||
`
|
`
|
||||||
<ul>
|
<ul>
|
||||||
<li>переход на Vue 3</li>
|
<li>переход на Vue 3</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
`
|
`
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
|
version: '0.10.3',
|
||||||
|
releaseDate: '2021-10-24',
|
||||||
showUntil: '2021-10-23',
|
showUntil: '2021-10-23',
|
||||||
header: '0.10.3 (2021-10-24)',
|
|
||||||
content:
|
content:
|
||||||
`
|
`
|
||||||
<ul>
|
<ul>
|
||||||
<li>исправления багов</li>
|
<li>исправления багов</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
`
|
`
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
|
version: '0.10.2',
|
||||||
|
releaseDate: '2021-10-19',
|
||||||
showUntil: '2021-10-18',
|
showUntil: '2021-10-18',
|
||||||
header: '0.10.2 (2021-10-19)',
|
|
||||||
content:
|
content:
|
||||||
`
|
`
|
||||||
<ul>
|
<ul>
|
||||||
<li>актуализация версий пакетов и стека используемых технологий</li>
|
<li>актуализация версий пакетов и стека используемых технологий</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
`
|
`
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
|
version: '0.10.1',
|
||||||
|
releaseDate: '2021-10-10',
|
||||||
showUntil: '2021-10-09',
|
showUntil: '2021-10-09',
|
||||||
header: '0.10.1 (2021-10-10)',
|
|
||||||
content:
|
content:
|
||||||
`
|
`
|
||||||
<ul>
|
<ul>
|
||||||
<li>исправления багов</li>
|
<li>исправления багов</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
`
|
`
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
|
version: '0.10.0',
|
||||||
|
releaseDate: '2021-02-09',
|
||||||
showUntil: '2021-02-16',
|
showUntil: '2021-02-16',
|
||||||
header: '0.10.0 (2021-02-09)',
|
|
||||||
content:
|
content:
|
||||||
`
|
`
|
||||||
<ul>
|
<ul>
|
||||||
@@ -65,12 +140,14 @@ export const versionHistory = [
|
|||||||
<li>в настройки добавлена возможность загрузки пользовательских обоев (пока без синхронизации)</li>
|
<li>в настройки добавлена возможность загрузки пользовательских обоев (пока без синхронизации)</li>
|
||||||
<li>немного улучшен парсинг fb2</li>
|
<li>немного улучшен парсинг fb2</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
`
|
`
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
|
version: '0.9.12',
|
||||||
|
releaseDate: '2020-12-18',
|
||||||
showUntil: '2020-12-17',
|
showUntil: '2020-12-17',
|
||||||
header: '0.9.12 (2020-12-18)',
|
|
||||||
content:
|
content:
|
||||||
`
|
`
|
||||||
<ul>
|
<ul>
|
||||||
@@ -79,23 +156,27 @@ export const versionHistory = [
|
|||||||
<li>добавлена кнопка для быстрого доступа к настройкам конвертирования</li>
|
<li>добавлена кнопка для быстрого доступа к настройкам конвертирования</li>
|
||||||
<li>улучшения работы конвертеров</li>
|
<li>улучшения работы конвертеров</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
`
|
`
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
|
version: '0.9.11',
|
||||||
|
releaseDate: '2020-12-09',
|
||||||
showUntil: '2020-12-08',
|
showUntil: '2020-12-08',
|
||||||
header: '0.9.11 (2020-12-09)',
|
|
||||||
content:
|
content:
|
||||||
`
|
`
|
||||||
<ul>
|
<ul>
|
||||||
<li>оптимизации, улучшения работы конвертеров</li>
|
<li>оптимизации, улучшения работы конвертеров</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
`
|
`
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
|
version: '0.9.10',
|
||||||
|
releaseDate: '2020-12-03',
|
||||||
showUntil: '2020-12-10',
|
showUntil: '2020-12-10',
|
||||||
header: '0.9.10 (2020-12-03)',
|
|
||||||
content:
|
content:
|
||||||
`
|
`
|
||||||
<ul>
|
<ul>
|
||||||
@@ -103,69 +184,81 @@ export const versionHistory = [
|
|||||||
<li>добавлена поддержка Rar-архивов</li>
|
<li>добавлена поддержка Rar-архивов</li>
|
||||||
<li>исправления багов</li>
|
<li>исправления багов</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
`
|
`
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
|
version: '0.9.9',
|
||||||
|
releaseDate: '2020-11-21',
|
||||||
showUntil: '2020-11-20',
|
showUntil: '2020-11-20',
|
||||||
header: '0.9.9 (2020-11-21)',
|
|
||||||
content:
|
content:
|
||||||
`
|
`
|
||||||
<ul>
|
<ul>
|
||||||
<li>оптимизации, исправления багов</li>
|
<li>оптимизации, исправления багов</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
`
|
`
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
|
version: '0.9.8',
|
||||||
|
releaseDate: '2020-11-13',
|
||||||
showUntil: '2020-11-12',
|
showUntil: '2020-11-12',
|
||||||
header: '0.9.8 (2020-11-13)',
|
|
||||||
content:
|
content:
|
||||||
`
|
`
|
||||||
<ul>
|
<ul>
|
||||||
<li>добавлено окно "Оглавление/закладки"</li>
|
<li>добавлено окно "Оглавление/закладки"</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
`
|
`
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
|
version: '0.9.7',
|
||||||
|
releaseDate: '2020-11-12',
|
||||||
showUntil: '2020-11-11',
|
showUntil: '2020-11-11',
|
||||||
header: '0.9.7 (2020-11-12)',
|
|
||||||
content:
|
content:
|
||||||
`
|
`
|
||||||
<ul>
|
<ul>
|
||||||
<li>исправления багов</li>
|
<li>исправления багов</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
`
|
`
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
|
version: '0.9.6',
|
||||||
|
releaseDate: '2020-11-06',
|
||||||
showUntil: '2020-11-05',
|
showUntil: '2020-11-05',
|
||||||
header: '0.9.6 (2020-11-06)',
|
|
||||||
content:
|
content:
|
||||||
`
|
`
|
||||||
<ul>
|
<ul>
|
||||||
<li>завершена работа над новым окном "Библиотека"</li>
|
<li>завершена работа над новым окном "Библиотека"</li>
|
||||||
<li>исправления багов</li>
|
<li>исправления багов</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
`
|
`
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
|
version: '0.9.5',
|
||||||
|
releaseDate: '2020-11-01',
|
||||||
showUntil: '2020-10-31',
|
showUntil: '2020-10-31',
|
||||||
header: '0.9.5 (2020-11-01)',
|
|
||||||
content:
|
content:
|
||||||
`
|
`
|
||||||
<ul>
|
<ul>
|
||||||
<li>на панель инструментов добавлена новая кнопка "Обновить с разбиением на параграфы"</li>
|
<li>на панель инструментов добавлена новая кнопка "Обновить с разбиением на параграфы"</li>
|
||||||
<li>исправления багов</li>
|
<li>исправления багов</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
`
|
`
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
|
version: '0.9.4',
|
||||||
|
releaseDate: '2020-10-29',
|
||||||
showUntil: '2020-10-28',
|
showUntil: '2020-10-28',
|
||||||
header: '0.9.4 (2020-10-29)',
|
|
||||||
content:
|
content:
|
||||||
`
|
`
|
||||||
<ul>
|
<ul>
|
||||||
@@ -173,23 +266,27 @@ export const versionHistory = [
|
|||||||
<li>для liberama.top добавлено новое окно: "Библиотека"</li>
|
<li>для liberama.top добавлено новое окно: "Библиотека"</li>
|
||||||
<li>исправления багов</li>
|
<li>исправления багов</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
`
|
`
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
|
version: '0.9.3',
|
||||||
|
releaseDate: '2020-05-21',
|
||||||
showUntil: '2020-05-20',
|
showUntil: '2020-05-20',
|
||||||
header: '0.9.3 (2020-05-21)',
|
|
||||||
content:
|
content:
|
||||||
`
|
`
|
||||||
<ul>
|
<ul>
|
||||||
<li>исправления багов</li>
|
<li>исправления багов</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
`
|
`
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
|
version: '0.9.2',
|
||||||
|
releaseDate: '2020-03-15',
|
||||||
showUntil: '2020-04-25',
|
showUntil: '2020-04-25',
|
||||||
header: '0.9.2 (2020-03-15)',
|
|
||||||
content:
|
content:
|
||||||
`
|
`
|
||||||
<ul>
|
<ul>
|
||||||
@@ -197,119 +294,139 @@ export const versionHistory = [
|
|||||||
<li>переход на Service Worker вместо AppCache для автономного режима работы</li>
|
<li>переход на Service Worker вместо AppCache для автономного режима работы</li>
|
||||||
<li>исправления багов</li>
|
<li>исправления багов</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
`
|
`
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
|
version: '0.9.1',
|
||||||
|
releaseDate: '2020-03-03',
|
||||||
showUntil: '2020-03-02',
|
showUntil: '2020-03-02',
|
||||||
header: '0.9.1 (2020-03-03)',
|
|
||||||
content:
|
content:
|
||||||
`
|
`
|
||||||
<ul>
|
<ul>
|
||||||
<li>улучшение работы серверной части</li>
|
<li>улучшение работы серверной части</li>
|
||||||
<li>незначительные изменения интерфейса</li>
|
<li>незначительные изменения интерфейса</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
`
|
`
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
|
version: '0.9.0',
|
||||||
|
releaseDate: '2020-02-26',
|
||||||
showUntil: '2020-02-25',
|
showUntil: '2020-02-25',
|
||||||
header: '0.9.0 (2020-02-26)',
|
|
||||||
content:
|
content:
|
||||||
`
|
`
|
||||||
<ul>
|
<ul>
|
||||||
<li>переход на UI-фреймфорк Quasar</li>
|
<li>переход на UI-фреймфорк Quasar</li>
|
||||||
<li>незначительные изменения интерфейса</li>
|
<li>незначительные изменения интерфейса</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
`
|
`
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
|
version: '0.8.4',
|
||||||
|
releaseDate: '2020-02-06',
|
||||||
showUntil: '2020-02-05',
|
showUntil: '2020-02-05',
|
||||||
header: '0.8.4 (2020-02-06)',
|
|
||||||
content:
|
content:
|
||||||
`
|
`
|
||||||
<ul>
|
<ul>
|
||||||
<li>добавлен paypal-адрес для пожертвований</li>
|
<li>добавлен paypal-адрес для пожертвований</li>
|
||||||
<li>исправления багов</li>
|
<li>исправления багов</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
`
|
`
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
|
version: '0.8.3',
|
||||||
|
releaseDate: '2020-01-28',
|
||||||
showUntil: '2020-01-27',
|
showUntil: '2020-01-27',
|
||||||
header: '0.8.3 (2020-01-28)',
|
|
||||||
content:
|
content:
|
||||||
`
|
`
|
||||||
<ul>
|
<ul>
|
||||||
<li>добавлено всплывающее окно с акцией "Оплатим хостинг вместе"</li>
|
<li>добавлено всплывающее окно с акцией "Оплатим хостинг вместе"</li>
|
||||||
<li>внутренние оптимизации</li>
|
<li>внутренние оптимизации</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
`
|
`
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
|
version: '0.8.2',
|
||||||
|
releaseDate: '2020-01-20',
|
||||||
showUntil: '2020-01-19',
|
showUntil: '2020-01-19',
|
||||||
header: '0.8.2 (2020-01-20)',
|
|
||||||
content:
|
content:
|
||||||
`
|
`
|
||||||
<ul>
|
<ul>
|
||||||
<li>внутренние оптимизации</li>
|
<li>внутренние оптимизации</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
`
|
`
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
|
version: '0.8.1',
|
||||||
|
releaseDate: '2020-01-07',
|
||||||
showUntil: '2020-01-06',
|
showUntil: '2020-01-06',
|
||||||
header: '0.8.1 (2020-01-07)',
|
|
||||||
content:
|
content:
|
||||||
`
|
`
|
||||||
<ul>
|
<ul>
|
||||||
<li>добавлена частичная поддержка формата FB3</li>
|
<li>добавлена частичная поддержка формата FB3</li>
|
||||||
<li>исправлен баг "Request path contains unescaped characters"</li>
|
<li>исправлен баг "Request path contains unescaped characters"</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
`
|
`
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
|
version: '0.8.0',
|
||||||
|
releaseDate: '2020-01-02',
|
||||||
showUntil: '2020-01-05',
|
showUntil: '2020-01-05',
|
||||||
header: '0.8.0 (2020-01-02)',
|
|
||||||
content:
|
content:
|
||||||
`
|
`
|
||||||
<ul>
|
<ul>
|
||||||
<li>окончательный переход на https</li>
|
<li>окончательный переход на https</li>
|
||||||
<li>код проекта теперь Open Source: <a href="https://github.com/bookpauk/liberama" target="_blank">https://github.com/bookpauk/liberama</a></li>
|
<li>код проекта теперь Open Source: <a href="https://github.com/bookpauk/liberama" target="_blank">https://github.com/bookpauk/liberama</a></li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
`
|
`
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
|
version: '0.7.9',
|
||||||
|
releaseDate: '2019-11-27',
|
||||||
showUntil: '2019-11-26',
|
showUntil: '2019-11-26',
|
||||||
header: '0.7.9 (2019-11-27)',
|
|
||||||
content:
|
content:
|
||||||
`
|
`
|
||||||
<ul>
|
<ul>
|
||||||
<li>добавлен неубираемый баннер для http-версии о переходе на httpS</li>
|
<li>добавлен неубираемый баннер для http-версии о переходе на httpS</li>
|
||||||
<li>исправления багов</li>
|
<li>исправления багов</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
`
|
`
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
|
version: '0.7.8',
|
||||||
|
releaseDate: '2019-11-25',
|
||||||
showUntil: '2019-11-24',
|
showUntil: '2019-11-24',
|
||||||
header: '0.7.8 (2019-11-25)',
|
|
||||||
content:
|
content:
|
||||||
`
|
`
|
||||||
<ul>
|
<ul>
|
||||||
<li>улучшение html-фильтров для сайтов</li>
|
<li>улучшение html-фильтров для сайтов</li>
|
||||||
<li>исправления багов</li>
|
<li>исправления багов</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
`
|
`
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
|
version: '0.7.7',
|
||||||
|
releaseDate: '2019-11-06',
|
||||||
showUntil: '2019-11-10',
|
showUntil: '2019-11-10',
|
||||||
header: '0.7.7 (2019-11-06)',
|
|
||||||
content:
|
content:
|
||||||
`
|
`
|
||||||
<ul>
|
<ul>
|
||||||
@@ -321,34 +438,40 @@ export const versionHistory = [
|
|||||||
<li style="list-style-type: square">от центра влево: уменьшить скорость скроллинга</li>
|
<li style="list-style-type: square">от центра влево: уменьшить скорость скроллинга</li>
|
||||||
</ul>
|
</ul>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
`
|
`
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
|
version: '0.7.6',
|
||||||
|
releaseDate: '2019-10-30',
|
||||||
showUntil: '2019-10-29',
|
showUntil: '2019-10-29',
|
||||||
header: '0.7.6 (2019-10-30)',
|
|
||||||
content:
|
content:
|
||||||
`
|
`
|
||||||
<ul>
|
<ul>
|
||||||
<li>исправления багов</li>
|
<li>исправления багов</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
`
|
`
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
|
version: '0.7.5',
|
||||||
|
releaseDate: '2019-10-22',
|
||||||
showUntil: '2019-10-21',
|
showUntil: '2019-10-21',
|
||||||
header: '0.7.5 (2019-10-22)',
|
|
||||||
content:
|
content:
|
||||||
`
|
`
|
||||||
<ul>
|
<ul>
|
||||||
<li>исправления багов</li>
|
<li>исправления багов</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
`
|
`
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
|
version: '0.7.3',
|
||||||
|
releaseDate: '2019-10-18',
|
||||||
showUntil: '2019-10-17',
|
showUntil: '2019-10-17',
|
||||||
header: '0.7.3 (2019-10-18)',
|
|
||||||
content:
|
content:
|
||||||
`
|
`
|
||||||
<ul>
|
<ul>
|
||||||
@@ -357,12 +480,14 @@ export const versionHistory = [
|
|||||||
<li>добавлен параметр "Включить html-фильтр для сайтов" в раздел "Вид"->"Текст" в настройках</li>
|
<li>добавлен параметр "Включить html-фильтр для сайтов" в раздел "Вид"->"Текст" в настройках</li>
|
||||||
<li>исправления багов</li>
|
<li>исправления багов</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
`
|
`
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
|
version: '0.7.1',
|
||||||
|
releaseDate: '2019-09-20',
|
||||||
showUntil: '2019-09-19',
|
showUntil: '2019-09-19',
|
||||||
header: '0.7.1 (2019-09-20)',
|
|
||||||
content:
|
content:
|
||||||
`
|
`
|
||||||
<ul>
|
<ul>
|
||||||
@@ -370,12 +495,14 @@ export const versionHistory = [
|
|||||||
<li>на панель управления добавлена кнопка "Автономный режим"</li>
|
<li>на панель управления добавлена кнопка "Автономный режим"</li>
|
||||||
<li>актуализирована справка</li>
|
<li>актуализирована справка</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
`
|
`
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
|
version: '0.7.0',
|
||||||
|
releaseDate: '2019-09-07',
|
||||||
showUntil: '2019-10-01',
|
showUntil: '2019-10-01',
|
||||||
header: '0.7.0 (2019-09-07)',
|
|
||||||
content:
|
content:
|
||||||
`
|
`
|
||||||
<ul>
|
<ul>
|
||||||
@@ -386,23 +513,27 @@ export const versionHistory = [
|
|||||||
<li>немного улучшен внешний вид и управление на смартфонах</li>
|
<li>немного улучшен внешний вид и управление на смартфонах</li>
|
||||||
<li>добавлен параметр "Компактность" в раздел "Вид"->"Текст" в настройках</li>
|
<li>добавлен параметр "Компактность" в раздел "Вид"->"Текст" в настройках</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
`
|
`
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
|
version: '0.6.10',
|
||||||
|
releaseDate: '2019-07-21',
|
||||||
showUntil: '2019-07-20',
|
showUntil: '2019-07-20',
|
||||||
header: '0.6.10 (2019-07-21)',
|
|
||||||
content:
|
content:
|
||||||
`
|
`
|
||||||
<ul>
|
<ul>
|
||||||
<li>исправления багов</li>
|
<li>исправления багов</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
`
|
`
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
|
version: '0.6.9',
|
||||||
|
releaseDate: '2019-06-23',
|
||||||
showUntil: '2019-06-22',
|
showUntil: '2019-06-22',
|
||||||
header: '0.6.9 (2019-06-23)',
|
|
||||||
content:
|
content:
|
||||||
`
|
`
|
||||||
<ul>
|
<ul>
|
||||||
@@ -413,12 +544,14 @@ export const versionHistory = [
|
|||||||
<li>улучшены прогрессбары</li>
|
<li>улучшены прогрессбары</li>
|
||||||
<li>исправления недочетов, небольшие оптимизации</li>
|
<li>исправления недочетов, небольшие оптимизации</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
`
|
`
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
|
version: '0.6.7',
|
||||||
|
releaseDate: '2019-05-30',
|
||||||
showUntil: '2019-06-05',
|
showUntil: '2019-06-05',
|
||||||
header: '0.6.7 (2019-05-30)',
|
|
||||||
content:
|
content:
|
||||||
`
|
`
|
||||||
<ul>
|
<ul>
|
||||||
@@ -431,36 +564,42 @@ export const versionHistory = [
|
|||||||
<li>добавлен GET-параметр вида "/reader?__pp=50.5&url=..." для указания позиции в книге в процентах</li>
|
<li>добавлен GET-параметр вида "/reader?__pp=50.5&url=..." для указания позиции в книге в процентах</li>
|
||||||
<li>исправления багов и недочетов</li>
|
<li>исправления багов и недочетов</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
`
|
`
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
|
version: '0.6.6',
|
||||||
|
releaseDate: '2019-03-29',
|
||||||
showUntil: '2019-03-29',
|
showUntil: '2019-03-29',
|
||||||
header: '0.6.6 (2019-03-29)',
|
|
||||||
content:
|
content:
|
||||||
`
|
`
|
||||||
<ul>
|
<ul>
|
||||||
<li>в справку добавлено описание настройки браузеров для автономной работы читалки (без доступа к интернету)</li>
|
<li>в справку добавлено описание настройки браузеров для автономной работы читалки (без доступа к интернету)</li>
|
||||||
<li>оптимизации процесса синхронизации, внутренние переделки</li>
|
<li>оптимизации процесса синхронизации, внутренние переделки</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
`
|
`
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
|
version: '0.6.4',
|
||||||
|
releaseDate: '2019-03-24',
|
||||||
showUntil: '2019-03-24',
|
showUntil: '2019-03-24',
|
||||||
header: '0.6.4 (2019-03-24)',
|
|
||||||
content:
|
content:
|
||||||
`
|
`
|
||||||
<ul>
|
<ul>
|
||||||
<li>исправления багов, оптимизации</li>
|
<li>исправления багов, оптимизации</li>
|
||||||
<li>добавлена возможность синхронизации данных между устройствами</li>
|
<li>добавлена возможность синхронизации данных между устройствами</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
`
|
`
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
|
version: '0.5.4',
|
||||||
|
releaseDate: '2019-03-04',
|
||||||
showUntil: '2019-03-04',
|
showUntil: '2019-03-04',
|
||||||
header: '0.5.4 (2019-03-04)',
|
|
||||||
content:
|
content:
|
||||||
`
|
`
|
||||||
<ul>
|
<ul>
|
||||||
@@ -469,12 +608,14 @@ export const versionHistory = [
|
|||||||
<li>(0.4.2) фильтр для СИ больше не вырезает изображения</li>
|
<li>(0.4.2) фильтр для СИ больше не вырезает изображения</li>
|
||||||
<li>(0.4.0) добавлено отображение картинок в fb2</li>
|
<li>(0.4.0) добавлено отображение картинок в fb2</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
`
|
`
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
|
version: '0.3.0',
|
||||||
|
releaseDate: '2019-02-17',
|
||||||
showUntil: '2019-02-17',
|
showUntil: '2019-02-17',
|
||||||
header: '0.3.0 (2019-02-17)',
|
|
||||||
content:
|
content:
|
||||||
`
|
`
|
||||||
<ul>
|
<ul>
|
||||||
@@ -482,12 +623,14 @@ export const versionHistory = [
|
|||||||
<li>улучшено распознавание текста</li>
|
<li>улучшено распознавание текста</li>
|
||||||
<li>изменена верстка страницы - убрано позиционирование каждого слова</li>
|
<li>изменена верстка страницы - убрано позиционирование каждого слова</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
`
|
`
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
|
version: '0.1.7',
|
||||||
|
releaseDate: '2019-02-14',
|
||||||
showUntil: '2019-02-14',
|
showUntil: '2019-02-14',
|
||||||
header: '0.1.7 (2019-02-14)',
|
|
||||||
content:
|
content:
|
||||||
`
|
`
|
||||||
<ul>
|
<ul>
|
||||||
@@ -497,17 +640,20 @@ export const versionHistory = [
|
|||||||
<li>добавлена возможность сброса настроек</li>
|
<li>добавлена возможность сброса настроек</li>
|
||||||
<li>убран автоматический редирект на последнюю загруженную книгу, если не задан url в маршруте</li>
|
<li>убран автоматический редирект на последнюю загруженную книгу, если не задан url в маршруте</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
`
|
`
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
|
version: '0.1.0',
|
||||||
|
releaseDate: '2019-02-12',
|
||||||
showUntil: '2019-02-12',
|
showUntil: '2019-02-12',
|
||||||
header: '0.1.0 (2019-02-12)',
|
|
||||||
content:
|
content:
|
||||||
`
|
`
|
||||||
<ul>
|
<ul>
|
||||||
<li>первый деплой проекта, длительность разработки - 2 месяца</li>
|
<li>первый деплой проекта, длительность разработки - 2 месяца</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
`
|
`
|
||||||
},
|
},
|
||||||
|
|
||||||
|
|||||||
@@ -55,6 +55,34 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<!--------------------------------------------------->
|
||||||
|
<div v-show="type == 'askYesNo'" class="bg-white no-wrap">
|
||||||
|
<div class="header row">
|
||||||
|
<div class="caption col row items-center q-ml-md">
|
||||||
|
<q-icon v-show="caption" class="q-mr-sm" :class="iconColor" :name="iconName" size="28px"></q-icon>
|
||||||
|
<div v-html="caption"></div>
|
||||||
|
</div>
|
||||||
|
<div class="close-icon column justify-center items-center">
|
||||||
|
<q-btn v-close-popup flat round dense>
|
||||||
|
<q-icon name="la la-times" size="18px"></q-icon>
|
||||||
|
</q-btn>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="q-mx-md">
|
||||||
|
<div v-html="message"></div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="buttons row justify-end q-pa-md">
|
||||||
|
<q-btn v-close-popup class="q-px-md q-ml-sm" dense no-caps>
|
||||||
|
Нет
|
||||||
|
</q-btn>
|
||||||
|
<q-btn class="q-px-md q-ml-sm" color="primary" dense no-caps @click="okClick">
|
||||||
|
Да
|
||||||
|
</q-btn>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
<!--------------------------------------------------->
|
<!--------------------------------------------------->
|
||||||
<div v-show="type == 'prompt'" class="bg-white no-wrap">
|
<div v-show="type == 'prompt'" class="bg-white no-wrap">
|
||||||
<div class="header row">
|
<div class="header row">
|
||||||
@@ -262,6 +290,23 @@ class StdDialog {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
askYesNo(message, caption, opts) {
|
||||||
|
return new Promise((resolve) => {
|
||||||
|
this.init(message, caption, opts);
|
||||||
|
|
||||||
|
this.hideTrigger = () => {
|
||||||
|
if (this.ok) {
|
||||||
|
resolve(true);
|
||||||
|
} else {
|
||||||
|
resolve(false);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
this.type = 'askYesNo';
|
||||||
|
this.active = true;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
prompt(message, caption, opts) {
|
prompt(message, caption, opts) {
|
||||||
return new Promise((resolve) => {
|
return new Promise((resolve) => {
|
||||||
this.enableValidator = false;
|
this.enableValidator = false;
|
||||||
|
|||||||
@@ -153,7 +153,7 @@ export default vueComponent(Window);
|
|||||||
}
|
}
|
||||||
|
|
||||||
.header {
|
.header {
|
||||||
background: linear-gradient(to bottom right, green, #59B04F);
|
background: linear-gradient(to bottom right, #007000, #59B04F);
|
||||||
align-items: center;
|
align-items: center;
|
||||||
height: 30px;
|
height: 30px;
|
||||||
}
|
}
|
||||||
@@ -161,8 +161,8 @@ export default vueComponent(Window);
|
|||||||
.header-text {
|
.header-text {
|
||||||
margin-left: 10px;
|
margin-left: 10px;
|
||||||
margin-right: 10px;
|
margin-right: 10px;
|
||||||
color: yellow;
|
color: #FFFFA0;
|
||||||
text-shadow: 2px 1px 5px black, 2px 2px 5px black;
|
text-shadow: 2px 2px 5px #005000, 2px 1px 5px #005000;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
white-space: nowrap;
|
white-space: nowrap;
|
||||||
}
|
}
|
||||||
@@ -174,7 +174,8 @@ export default vueComponent(Window);
|
|||||||
}
|
}
|
||||||
|
|
||||||
.close-button:hover {
|
.close-button:hover {
|
||||||
background-color: #69C05F;
|
color: white;
|
||||||
|
background-color: #FF3030;
|
||||||
}
|
}
|
||||||
|
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -32,6 +32,8 @@ import {QPopupProxy} from 'quasar/src/components/popup-proxy';
|
|||||||
import {QDialog} from 'quasar/src/components/dialog';
|
import {QDialog} from 'quasar/src/components/dialog';
|
||||||
import {QChip} from 'quasar/src/components/chip';
|
import {QChip} from 'quasar/src/components/chip';
|
||||||
import {QTree} from 'quasar/src/components/tree';
|
import {QTree} from 'quasar/src/components/tree';
|
||||||
|
import {QVirtualScroll} from 'quasar/src/components/virtual-scroll';
|
||||||
|
|
||||||
//import {QExpansionItem} from 'quasar/src/components/expansion-item';
|
//import {QExpansionItem} from 'quasar/src/components/expansion-item';
|
||||||
|
|
||||||
const components = {
|
const components = {
|
||||||
@@ -62,6 +64,7 @@ const components = {
|
|||||||
QChip,
|
QChip,
|
||||||
QTree,
|
QTree,
|
||||||
//QExpansionItem,
|
//QExpansionItem,
|
||||||
|
QVirtualScroll,
|
||||||
};
|
};
|
||||||
|
|
||||||
//directives
|
//directives
|
||||||
@@ -86,7 +89,6 @@ const plugins = {
|
|||||||
import '@quasar/extras/line-awesome/line-awesome.css';
|
import '@quasar/extras/line-awesome/line-awesome.css';
|
||||||
import lineAwesome from 'quasar/icon-set/line-awesome.js'
|
import lineAwesome from 'quasar/icon-set/line-awesome.js'
|
||||||
|
|
||||||
//const q: {Quasar, QuasarOptions: { config, components, directives, plugins }};
|
|
||||||
export default {
|
export default {
|
||||||
quasar: Quasar,
|
quasar: Quasar,
|
||||||
options: { config, components, directives, plugins },
|
options: { config, components, directives, plugins },
|
||||||
|
|||||||
53
client/share/LockQueue.js
Normal file
53
client/share/LockQueue.js
Normal file
@@ -0,0 +1,53 @@
|
|||||||
|
class LockQueue {
|
||||||
|
constructor(queueSize) {
|
||||||
|
this.queueSize = queueSize;
|
||||||
|
this.freed = true;
|
||||||
|
this.waitingQueue = [];
|
||||||
|
}
|
||||||
|
|
||||||
|
//async
|
||||||
|
get(take = true) {
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
if (this.freed) {
|
||||||
|
if (take)
|
||||||
|
this.freed = false;
|
||||||
|
resolve();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.waitingQueue.length < this.queueSize) {
|
||||||
|
this.waitingQueue.push({resolve, reject});
|
||||||
|
} else {
|
||||||
|
reject(new Error('Lock queue is too long'));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
ret() {
|
||||||
|
if (this.waitingQueue.length) {
|
||||||
|
this.waitingQueue.shift().resolve();
|
||||||
|
} else {
|
||||||
|
this.freed = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//async
|
||||||
|
wait() {
|
||||||
|
return this.get(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
retAll() {
|
||||||
|
while (this.waitingQueue.length) {
|
||||||
|
this.waitingQueue.shift().resolve();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
errAll(error = 'rejected') {
|
||||||
|
while (this.waitingQueue.length) {
|
||||||
|
this.waitingQueue.shift().reject(new Error(error));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
export default LockQueue;
|
||||||
@@ -90,7 +90,7 @@ export function toBase58(data) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export function fromBase58(data) {
|
export function fromBase58(data) {
|
||||||
return bs58.decode(data);
|
return Buffer.from(bs58.decode(data));
|
||||||
}
|
}
|
||||||
|
|
||||||
//base-x слишком тормозит, используем sjcl
|
//base-x слишком тормозит, используем sjcl
|
||||||
@@ -107,6 +107,10 @@ export function fromBase64(data) {
|
|||||||
));
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function hasProp(obj, prop) {
|
||||||
|
return Object.prototype.hasOwnProperty.call(obj, prop);
|
||||||
|
}
|
||||||
|
|
||||||
export function getObjDiff(oldObj, newObj, opts = {}) {
|
export function getObjDiff(oldObj, newObj, opts = {}) {
|
||||||
const {
|
const {
|
||||||
exclude = [],
|
exclude = [],
|
||||||
@@ -126,7 +130,7 @@ export function getObjDiff(oldObj, newObj, opts = {}) {
|
|||||||
for (const key of Object.keys(oldObj)) {
|
for (const key of Object.keys(oldObj)) {
|
||||||
const kp = `${keyPath}${key}`;
|
const kp = `${keyPath}${key}`;
|
||||||
|
|
||||||
if (newObj.hasOwnProperty(key)) {
|
if (Object.prototype.hasOwnProperty.call(newObj, key)) {
|
||||||
if (ex.has(kp))
|
if (ex.has(kp))
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
@@ -149,7 +153,7 @@ export function getObjDiff(oldObj, newObj, opts = {}) {
|
|||||||
if (exAdd.has(kp))
|
if (exAdd.has(kp))
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
if (!oldObj.hasOwnProperty(key)) {
|
if (!Object.prototype.hasOwnProperty.call(oldObj, key)) {
|
||||||
result.add[key] = _.cloneDeep(newObj[key]);
|
result.add[key] = _.cloneDeep(newObj[key]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -213,7 +217,7 @@ export function applyObjDiff(obj, diff, opts = {}) {
|
|||||||
|
|
||||||
const change = diff.change;
|
const change = diff.change;
|
||||||
for (const key of Object.keys(change)) {
|
for (const key of Object.keys(change)) {
|
||||||
if (result.hasOwnProperty(key)) {
|
if (Object.prototype.hasOwnProperty.call(result, key)) {
|
||||||
if (_.isObject(change[key])) {
|
if (_.isObject(change[key])) {
|
||||||
result[key] = applyObjDiff(result[key], change[key], opts);
|
result[key] = applyObjDiff(result[key], change[key], opts);
|
||||||
} else {
|
} else {
|
||||||
@@ -359,4 +363,50 @@ 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); });
|
||||||
}
|
}
|
||||||
@@ -2,8 +2,10 @@ import * as utils from '../../share/utils';
|
|||||||
import googleFonts from './fonts/fonts.json';
|
import googleFonts from './fonts/fonts.json';
|
||||||
|
|
||||||
const readerActions = {
|
const readerActions = {
|
||||||
'help': 'Вызвать cправку',
|
|
||||||
'loader': 'На страницу загрузки',
|
'loader': 'На страницу загрузки',
|
||||||
|
'loadFile': 'Загрузить файл с диска',
|
||||||
|
'loadBuffer': 'Загрузить из буфера обмена',
|
||||||
|
'help': 'Вызвать cправку',
|
||||||
'settings': 'Настроить',
|
'settings': 'Настроить',
|
||||||
'undoAction': 'Действие назад',
|
'undoAction': 'Действие назад',
|
||||||
'redoAction': 'Действие вперед',
|
'redoAction': 'Действие вперед',
|
||||||
@@ -15,10 +17,11 @@ const readerActions = {
|
|||||||
'copyText': 'Скопировать текст со страницы',
|
'copyText': 'Скопировать текст со страницы',
|
||||||
'convOptions': 'Настроить конвертирование',
|
'convOptions': 'Настроить конвертирование',
|
||||||
'refresh': 'Принудительно обновить книгу',
|
'refresh': 'Принудительно обновить книгу',
|
||||||
|
'clickControl': 'Управление кликом',
|
||||||
'offlineMode': 'Автономный режим (без интернета)',
|
'offlineMode': 'Автономный режим (без интернета)',
|
||||||
'contents': 'Оглавление/закладки',
|
'contents': 'Оглавление/закладки',
|
||||||
'libs': 'Сетевая библиотека',
|
'libs': 'Сетевая библиотека',
|
||||||
'recentBooks': 'Открыть недавние',
|
'recentBooks': 'Показать загруженные',
|
||||||
'switchToolbar': 'Показать/скрыть панель управления',
|
'switchToolbar': 'Показать/скрыть панель управления',
|
||||||
'donate': '',
|
'donate': '',
|
||||||
'bookBegin': 'В начало книги',
|
'bookBegin': 'В начало книги',
|
||||||
@@ -35,6 +38,9 @@ const readerActions = {
|
|||||||
|
|
||||||
//readerActions[name]
|
//readerActions[name]
|
||||||
const toolButtons = [
|
const toolButtons = [
|
||||||
|
{name: 'loadFile', show: true},
|
||||||
|
{name: 'loadBuffer', show: true},
|
||||||
|
{name: 'help', show: true},
|
||||||
{name: 'undoAction', show: true},
|
{name: 'undoAction', show: true},
|
||||||
{name: 'redoAction', show: true},
|
{name: 'redoAction', show: true},
|
||||||
{name: 'fullScreen', show: true},
|
{name: 'fullScreen', show: true},
|
||||||
@@ -47,13 +53,16 @@ const toolButtons = [
|
|||||||
{name: 'contents', show: true},
|
{name: 'contents', show: true},
|
||||||
{name: 'libs', show: true},
|
{name: 'libs', show: true},
|
||||||
{name: 'recentBooks', show: true},
|
{name: 'recentBooks', show: true},
|
||||||
|
{name: 'clickControl', show: false},
|
||||||
{name: 'offlineMode', show: false},
|
{name: 'offlineMode', show: false},
|
||||||
];
|
];
|
||||||
|
|
||||||
//readerActions[name]
|
//readerActions[name]
|
||||||
const hotKeys = [
|
const hotKeys = [
|
||||||
{name: 'help', codes: ['F1', 'H']},
|
|
||||||
{name: 'loader', codes: ['Escape']},
|
{name: 'loader', codes: ['Escape']},
|
||||||
|
{name: 'loadFile', codes: ['F3']},
|
||||||
|
{name: 'loadBuffer', codes: ['F4']},
|
||||||
|
{name: 'help', codes: ['F1', 'H']},
|
||||||
{name: 'settings', codes: ['S']},
|
{name: 'settings', codes: ['S']},
|
||||||
{name: 'undoAction', codes: ['Ctrl+BracketLeft']},
|
{name: 'undoAction', codes: ['Ctrl+BracketLeft']},
|
||||||
{name: 'redoAction', codes: ['Ctrl+BracketRight']},
|
{name: 'redoAction', codes: ['Ctrl+BracketRight']},
|
||||||
@@ -61,12 +70,13 @@ const hotKeys = [
|
|||||||
{name: 'scrolling', codes: ['Z']},
|
{name: 'scrolling', codes: ['Z']},
|
||||||
{name: 'setPosition', codes: ['P']},
|
{name: 'setPosition', codes: ['P']},
|
||||||
{name: 'search', codes: ['Ctrl+F']},
|
{name: 'search', codes: ['Ctrl+F']},
|
||||||
{name: 'copyText', codes: ['Ctrl+C']},
|
{name: 'copyText', codes: ['Ctrl+Space']},
|
||||||
{name: 'convOptions', codes: ['Ctrl+M']},
|
{name: 'convOptions', codes: ['Ctrl+M']},
|
||||||
{name: 'refresh', codes: ['R']},
|
{name: 'refresh', codes: ['R']},
|
||||||
{name: 'contents', codes: ['C']},
|
{name: 'contents', codes: ['C']},
|
||||||
{name: 'libs', codes: ['L']},
|
{name: 'libs', codes: ['L']},
|
||||||
{name: 'recentBooks', codes: ['X']},
|
{name: 'recentBooks', codes: ['X']},
|
||||||
|
{name: 'clickControl', codes: ['Ctrl+B']},
|
||||||
{name: 'offlineMode', codes: ['O']},
|
{name: 'offlineMode', codes: ['O']},
|
||||||
|
|
||||||
{name: 'switchToolbar', codes: ['Tab', 'Q']},
|
{name: 'switchToolbar', codes: ['Tab', 'Q']},
|
||||||
@@ -175,8 +185,14 @@ const settingDefaults = {
|
|||||||
|
|
||||||
fontShifts: {},
|
fontShifts: {},
|
||||||
showToolButton: {},
|
showToolButton: {},
|
||||||
|
toolBarHideOnScroll: true,
|
||||||
userHotKeys: {},
|
userHotKeys: {},
|
||||||
userWallpapers: [],
|
userWallpapers: [],
|
||||||
|
|
||||||
|
recentShowSameBook: false,
|
||||||
|
recentSortMethod: '',
|
||||||
|
|
||||||
|
needUpdateSettingsView: 0,
|
||||||
};
|
};
|
||||||
|
|
||||||
for (const font of fonts)
|
for (const font of fonts)
|
||||||
@@ -212,9 +228,6 @@ const libsDefaults = {
|
|||||||
{r: 'http://flibusta.is', s: 'http://flibusta.is', list: [
|
{r: 'http://flibusta.is', s: 'http://flibusta.is', list: [
|
||||||
{l: 'http://flibusta.is', c: 'Флибуста | Книжное братство'},
|
{l: 'http://flibusta.is', c: 'Флибуста | Книжное братство'},
|
||||||
]},
|
]},
|
||||||
{r: 'https://flibs.in', s: 'https://flibs.in', list: [
|
|
||||||
{l: 'https://flibs.in', c: 'Flibs'},
|
|
||||||
]},
|
|
||||||
{r: 'http://fantasy-worlds.org', s: 'http://fantasy-worlds.org', list: [
|
{r: 'http://fantasy-worlds.org', s: 'http://fantasy-worlds.org', list: [
|
||||||
{l: 'http://fantasy-worlds.org', c: 'Миры Фэнтези'},
|
{l: 'http://fantasy-worlds.org', c: 'Миры Фэнтези'},
|
||||||
]},
|
]},
|
||||||
|
|||||||
@@ -18,7 +18,7 @@ server {
|
|||||||
|
|
||||||
server_name liberama.top;
|
server_name liberama.top;
|
||||||
|
|
||||||
client_max_body_size 50m;
|
client_max_body_size 100m;
|
||||||
proxy_read_timeout 1h;
|
proxy_read_timeout 1h;
|
||||||
|
|
||||||
gzip on;
|
gzip on;
|
||||||
@@ -35,6 +35,7 @@ server {
|
|||||||
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 / {
|
||||||
@@ -62,7 +63,7 @@ server {
|
|||||||
listen 80;
|
listen 80;
|
||||||
server_name b.liberama.top;
|
server_name b.liberama.top;
|
||||||
|
|
||||||
client_max_body_size 50m;
|
client_max_body_size 100m;
|
||||||
proxy_read_timeout 1h;
|
proxy_read_timeout 1h;
|
||||||
|
|
||||||
gzip on;
|
gzip on;
|
||||||
@@ -139,5 +140,6 @@ server {
|
|||||||
|
|
||||||
location / {
|
location / {
|
||||||
proxy_pass http://fantasy-worlds.org;
|
proxy_pass http://fantasy-worlds.org;
|
||||||
|
proxy_hide_header x-frame-options;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
#!/bin/bash
|
#!/bin/bash
|
||||||
|
|
||||||
if ! pgrep -x "liberama" > /dev/null ; then
|
if ! pgrep -x "liberama" > /dev/null ; then
|
||||||
sudo -H -u www-data bash -c "cd /var/www; /home/liberama/liberama"
|
sudo -H -u www-data bash -c "cd /var/www; /home/liberama/liberama >/dev/null"
|
||||||
else
|
else
|
||||||
echo "Process 'liberama' already running"
|
echo "Process 'liberama' already running"
|
||||||
fi
|
fi
|
||||||
|
|||||||
@@ -7,7 +7,7 @@ server {
|
|||||||
|
|
||||||
server_name omnireader.ru;
|
server_name omnireader.ru;
|
||||||
|
|
||||||
client_max_body_size 50m;
|
client_max_body_size 100m;
|
||||||
proxy_read_timeout 1h;
|
proxy_read_timeout 1h;
|
||||||
|
|
||||||
gzip on;
|
gzip on;
|
||||||
@@ -24,6 +24,7 @@ server {
|
|||||||
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 / {
|
||||||
@@ -51,7 +52,7 @@ server {
|
|||||||
listen 80;
|
listen 80;
|
||||||
server_name old.omnireader.ru;
|
server_name old.omnireader.ru;
|
||||||
|
|
||||||
client_max_body_size 50m;
|
client_max_body_size 100m;
|
||||||
|
|
||||||
gzip on;
|
gzip on;
|
||||||
gzip_min_length 1024;
|
gzip_min_length 1024;
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
#!/bin/bash
|
#!/bin/bash
|
||||||
|
|
||||||
sudo -H -u www-data bash -c "cd /var/www; /home/liberama/liberama" & disown
|
sudo -H -u www-data bash -c "cd /var/www; /home/liberama/liberama >/dev/null & disown"
|
||||||
sudo service cron start
|
sudo service cron start
|
||||||
|
|||||||
12125
package-lock.json
generated
12125
package-lock.json
generated
File diff suppressed because it is too large
Load Diff
36
package.json
36
package.json
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "Liberama",
|
"name": "Liberama",
|
||||||
"version": "0.11.1",
|
"version": "0.11.8",
|
||||||
"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",
|
||||||
@@ -10,8 +10,8 @@
|
|||||||
"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 -o dist/linux/liberama .",
|
"build:linux": "npm run build:client && node build/linux && pkg -t node14-linux-x64 -C GZip -o dist/linux/liberama .",
|
||||||
"build:win": "npm run build:client && node build/win && pkg -t node14-win-x64 -o dist/win/liberama .",
|
"build:win": "npm run build:client && node build/win && pkg -t node14-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"
|
||||||
@@ -28,16 +28,17 @@
|
|||||||
"@babel/preset-env": "^7.16.0",
|
"@babel/preset-env": "^7.16.0",
|
||||||
"@vue/compiler-sfc": "^3.2.22",
|
"@vue/compiler-sfc": "^3.2.22",
|
||||||
"babel-loader": "^8.2.3",
|
"babel-loader": "^8.2.3",
|
||||||
"copy-webpack-plugin": "^9.1.0",
|
"copy-webpack-plugin": "^11.0.0",
|
||||||
"css-loader": "^6.5.1",
|
"css-loader": "^6.5.1",
|
||||||
"css-minimizer-webpack-plugin": "^3.1.3",
|
"css-minimizer-webpack-plugin": "^4.0.0",
|
||||||
"eslint": "^8.2.0",
|
"eslint": "^8.19.0",
|
||||||
"eslint-plugin-vue": "^8.0.3",
|
"eslint-plugin-vue": "^9.2.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.4.4",
|
||||||
|
"pkg": "^5.5.1",
|
||||||
"terser-webpack-plugin": "^5.2.5",
|
"terser-webpack-plugin": "^5.2.5",
|
||||||
"vue-eslint-parser": "^8.0.1",
|
"vue-eslint-parser": "^9.0.3",
|
||||||
"vue-loader": "^16.8.3",
|
"vue-loader": "^17.0.0",
|
||||||
"vue-style-loader": "^4.1.3",
|
"vue-style-loader": "^4.1.3",
|
||||||
"webpack": "^5.64.1",
|
"webpack": "^5.64.1",
|
||||||
"webpack-cli": "^4.9.1",
|
"webpack-cli": "^4.9.1",
|
||||||
@@ -49,25 +50,24 @@
|
|||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@quasar/extras": "^1.12.0",
|
"@quasar/extras": "^1.12.0",
|
||||||
"@vue/compat": "^3.2.21",
|
"@vue/compat": "^3.2.21",
|
||||||
"axios": "^0.24.0",
|
"axios": "^0.27.2",
|
||||||
"base-x": "^3.0.9",
|
"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.17.1",
|
||||||
"fg-loadcss": "^3.1.0",
|
"fg-loadcss": "^3.1.0",
|
||||||
"fs-extra": "^9.0.1",
|
"fs-extra": "^10.1.0",
|
||||||
"got": "^11.8.2",
|
|
||||||
"he": "^1.2.0",
|
"he": "^1.2.0",
|
||||||
"iconv-lite": "^0.6.3",
|
"iconv-lite": "^0.6.3",
|
||||||
|
"jembadb": "^3.0.8",
|
||||||
"localforage": "^1.10.0",
|
"localforage": "^1.10.0",
|
||||||
"lodash": "^4.17.21",
|
"lodash": "^4.17.21",
|
||||||
"minimist": "^1.2.5",
|
"minimist": "^1.2.5",
|
||||||
"multer": "^1.4.3",
|
"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",
|
||||||
"pkg": "^4.4.9",
|
"quasar": "^2.7.5",
|
||||||
"quasar": "^2.3.2",
|
|
||||||
"safe-buffer": "^5.2.1",
|
"safe-buffer": "^5.2.1",
|
||||||
"sanitize-html": "^2.5.3",
|
"sanitize-html": "^2.5.3",
|
||||||
"sjcl": "^1.0.8",
|
"sjcl": "^1.0.8",
|
||||||
@@ -76,8 +76,8 @@
|
|||||||
"sqlite3": "^5.0.2",
|
"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.22",
|
"vue": "^3.2.37",
|
||||||
"vue-router": "^4.0.12",
|
"vue-router": "^4.1.1",
|
||||||
"vuex": "^4.0.2",
|
"vuex": "^4.0.2",
|
||||||
"vuex-persistedstate": "^4.1.0",
|
"vuex-persistedstate": "^4.1.0",
|
||||||
"webdav": "^4.7.0",
|
"webdav": "^4.7.0",
|
||||||
|
|||||||
@@ -22,7 +22,8 @@ module.exports = {
|
|||||||
maxUploadPublicDirSize: 200*1024*1024,//100Мб
|
maxUploadPublicDirSize: 200*1024*1024,//100Мб
|
||||||
|
|
||||||
useExternalBookConverter: false,
|
useExternalBookConverter: false,
|
||||||
webConfigParams: ['name', 'version', 'mode', 'maxUploadFileSize', 'useExternalBookConverter', 'branch'],
|
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'],
|
||||||
|
|
||||||
db: [
|
db: [
|
||||||
{
|
{
|
||||||
@@ -48,7 +49,7 @@ module.exports = {
|
|||||||
servers: [
|
servers: [
|
||||||
{
|
{
|
||||||
serverName: '1',
|
serverName: '1',
|
||||||
mode: 'normal', //'none', 'normal', 'site', 'reader', 'omnireader'
|
mode: 'normal', //'none', 'normal', 'site', 'reader', 'omnireader', 'liberama.top'
|
||||||
ip: '0.0.0.0',
|
ip: '0.0.0.0',
|
||||||
port: '33080',
|
port: '33080',
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -25,6 +25,10 @@ class WebSocketController {
|
|||||||
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);
|
||||||
@@ -55,8 +59,7 @@ class WebSocketController {
|
|||||||
ws.lastActivity = Date.now();
|
ws.lastActivity = Date.now();
|
||||||
|
|
||||||
//pong for WebSocketConnection
|
//pong for WebSocketConnection
|
||||||
if (req._rpo === 1)
|
this.send({_rok: 1}, req, ws);
|
||||||
this.send({_rok: 1}, req, ws);
|
|
||||||
|
|
||||||
switch (req.action) {
|
switch (req.action) {
|
||||||
case 'test':
|
case 'test':
|
||||||
@@ -71,6 +74,10 @@ class WebSocketController {
|
|||||||
await this.readerRestoreCachedFile(req, ws); break;
|
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;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
throw new Error(`Action not found: ${req.action}`);
|
throw new Error(`Action not found: ${req.action}`);
|
||||||
@@ -169,6 +176,20 @@ 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);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
module.exports = WebSocketController;
|
module.exports = WebSocketController;
|
||||||
|
|||||||
@@ -25,60 +25,7 @@ class WorkerController extends BaseController {
|
|||||||
res.status(400).send({error});
|
res.status(400).send({error});
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
//TODO: удалить бесполезную getStateFinish
|
|
||||||
async getStateFinish(req, res) {
|
|
||||||
const request = req.body;
|
|
||||||
let error = '';
|
|
||||||
try {
|
|
||||||
if (!request.workerId)
|
|
||||||
throw new Error(`key 'workerId' is wrong`);
|
|
||||||
|
|
||||||
res.writeHead(200, {
|
|
||||||
'Content-Type': 'text/json; charset=utf-8',
|
|
||||||
});
|
|
||||||
|
|
||||||
const splitter = '-- aod2t5hDXU32bUFyqlFE next status --';
|
|
||||||
const refreshPause = 200;
|
|
||||||
let i = 0;
|
|
||||||
let prevProgress = -1;
|
|
||||||
let prevState = '';
|
|
||||||
let state;
|
|
||||||
while (1) {// eslint-disable-line no-constant-condition
|
|
||||||
state = this.workerState.getState(request.workerId);
|
|
||||||
if (!state) break;
|
|
||||||
|
|
||||||
res.write(splitter + JSON.stringify(state));
|
|
||||||
res.flush();
|
|
||||||
|
|
||||||
if (state.state != 'finish' && state.state != 'error')
|
|
||||||
await utils.sleep(refreshPause);
|
|
||||||
else
|
|
||||||
break;
|
|
||||||
|
|
||||||
i++;
|
|
||||||
if (i > 2*60*1000/refreshPause) {//2 мин ждем телодвижений воркера
|
|
||||||
res.write(splitter + JSON.stringify({state: 'error', error: 'Слишком долгое время ожидания'}));
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
i = (prevProgress != state.progress || prevState != state.state ? 1 : i);
|
|
||||||
prevProgress = state.progress;
|
|
||||||
prevState = state.state;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!state) {
|
|
||||||
res.write(splitter + JSON.stringify({}));
|
|
||||||
}
|
|
||||||
|
|
||||||
res.end();
|
|
||||||
return false;
|
|
||||||
} catch (e) {
|
|
||||||
error = e.message;
|
|
||||||
}
|
|
||||||
//bad request
|
|
||||||
res.status(400).send({error});
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
module.exports = WorkerController;
|
module.exports = WorkerController;
|
||||||
|
|||||||
@@ -5,30 +5,26 @@ const exitSignals = ['SIGINT', 'SIGTERM', 'SIGBREAK', 'SIGHUP', 'uncaughtExcepti
|
|||||||
|
|
||||||
//singleton
|
//singleton
|
||||||
class AsyncExit {
|
class AsyncExit {
|
||||||
constructor() {
|
constructor(signals = exitSignals, codeOnSignal = 2) {
|
||||||
if (!instance) {
|
if (!instance) {
|
||||||
this.onSignalCallbacks = new Map();
|
this.onSignalCallbacks = new Map();
|
||||||
this.callbacks = new Map();
|
this.callbacks = new Map();
|
||||||
this.afterCallbacks = new Map();
|
this.afterCallbacks = new Map();
|
||||||
this.exitTimeout = defaultTimeout;
|
this.exitTimeout = defaultTimeout;
|
||||||
this.inited = false;
|
|
||||||
|
this._init(signals, codeOnSignal);
|
||||||
|
|
||||||
instance = this;
|
instance = this;
|
||||||
}
|
}
|
||||||
|
|
||||||
return instance;
|
return instance;
|
||||||
}
|
}
|
||||||
|
|
||||||
init(signals = null, codeOnSignal = 2) {
|
_init(signals, codeOnSignal) {
|
||||||
if (this.inited)
|
const runSingalCallbacks = async(signal, err, origin) => {
|
||||||
throw new Error('AsyncExit: initialized already');
|
|
||||||
|
|
||||||
if (!signals)
|
|
||||||
signals = exitSignals;
|
|
||||||
|
|
||||||
const runSingalCallbacks = async(signal) => {
|
|
||||||
for (const signalCallback of this.onSignalCallbacks.keys()) {
|
for (const signalCallback of this.onSignalCallbacks.keys()) {
|
||||||
try {
|
try {
|
||||||
await signalCallback(signal);
|
await signalCallback(signal, err, origin);
|
||||||
} catch(e) {
|
} catch(e) {
|
||||||
console.error(e);
|
console.error(e);
|
||||||
}
|
}
|
||||||
@@ -36,13 +32,11 @@ class AsyncExit {
|
|||||||
};
|
};
|
||||||
|
|
||||||
for (const signal of signals) {
|
for (const signal of signals) {
|
||||||
process.once(signal, async() => {
|
process.once(signal, async(err, origin) => {
|
||||||
await runSingalCallbacks(signal);
|
await runSingalCallbacks(signal, err, origin);
|
||||||
this.exit(codeOnSignal);
|
this.exit(codeOnSignal);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
this.inited = true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
onSignal(signalCallback) {
|
onSignal(signalCallback) {
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
const got = require('got');
|
const axios = require('axios');
|
||||||
|
|
||||||
class FileDownloader {
|
class FileDownloader {
|
||||||
constructor(limitDownloadSize = 0) {
|
constructor(limitDownloadSize = 0) {
|
||||||
@@ -7,54 +7,82 @@ class FileDownloader {
|
|||||||
|
|
||||||
async load(url, callback, abort) {
|
async load(url, callback, abort) {
|
||||||
let errMes = '';
|
let errMes = '';
|
||||||
|
|
||||||
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': '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'
|
||||||
},
|
},
|
||||||
responseType: 'buffer',
|
responseType: 'stream',
|
||||||
};
|
};
|
||||||
|
|
||||||
const response = await got(url, Object.assign({}, options, {method: 'HEAD'}));
|
|
||||||
|
|
||||||
let estSize = 0;
|
|
||||||
if (response.headers['content-length']) {
|
|
||||||
estSize = response.headers['content-length'];
|
|
||||||
}
|
|
||||||
|
|
||||||
let prevProg = 0;
|
|
||||||
const request = got(url, options);
|
|
||||||
|
|
||||||
request.on('downloadProgress', progress => {
|
|
||||||
if (this.limitDownloadSize) {
|
|
||||||
if (progress.transferred > this.limitDownloadSize) {
|
|
||||||
errMes = 'Файл слишком большой';
|
|
||||||
request.cancel();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
let prog = 0;
|
|
||||||
if (estSize)
|
|
||||||
prog = Math.round(progress.transferred/estSize*100);
|
|
||||||
else if (progress.transferred)
|
|
||||||
prog = Math.round(progress.transferred/(progress.transferred + 200000)*100);
|
|
||||||
|
|
||||||
if (prog != prevProg && callback)
|
|
||||||
callback(prog);
|
|
||||||
prevProg = prog;
|
|
||||||
|
|
||||||
if (abort && abort()) {
|
|
||||||
errMes = 'abort';
|
|
||||||
request.cancel();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
try {
|
try {
|
||||||
return (await request).body;
|
const res = await axios.get(url, options);
|
||||||
|
|
||||||
|
let estSize = 0;
|
||||||
|
if (res.headers['content-length']) {
|
||||||
|
estSize = res.headers['content-length'];
|
||||||
|
}
|
||||||
|
|
||||||
|
if (estSize > this.limitDownloadSize) {
|
||||||
|
throw new Error('Файл слишком большой');
|
||||||
|
}
|
||||||
|
|
||||||
|
let prevProg = 0;
|
||||||
|
let transferred = 0;
|
||||||
|
|
||||||
|
const download = this.streamToBuffer(res.data, (chunk) => {
|
||||||
|
transferred += chunk.length;
|
||||||
|
if (this.limitDownloadSize) {
|
||||||
|
if (transferred > this.limitDownloadSize) {
|
||||||
|
errMes = 'Файл слишком большой';
|
||||||
|
res.request.abort();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let prog = 0;
|
||||||
|
if (estSize)
|
||||||
|
prog = Math.round(transferred/estSize*100);
|
||||||
|
else
|
||||||
|
prog = Math.round(transferred/(transferred + 200000)*100);
|
||||||
|
|
||||||
|
if (prog != prevProg && callback)
|
||||||
|
callback(prog);
|
||||||
|
prevProg = prog;
|
||||||
|
|
||||||
|
if (abort && abort()) {
|
||||||
|
errMes = 'abort';
|
||||||
|
res.request.abort();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return await download;
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
errMes = (errMes ? errMes : error.message);
|
errMes = (errMes ? errMes : error.message);
|
||||||
throw new Error(errMes);
|
throw new Error(errMes);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
streamToBuffer(stream, progress) {
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
|
||||||
|
if (!progress)
|
||||||
|
progress = () => {};
|
||||||
|
|
||||||
|
const _buf = [];
|
||||||
|
|
||||||
|
stream.on('data', (chunk) => {
|
||||||
|
_buf.push(chunk);
|
||||||
|
progress(chunk);
|
||||||
|
});
|
||||||
|
stream.on('end', () => resolve(Buffer.concat(_buf)));
|
||||||
|
stream.on('error', (err) => {
|
||||||
|
reject(err);
|
||||||
|
});
|
||||||
|
stream.on('aborted', () => {
|
||||||
|
reject(new Error('aborted'));
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
module.exports = FileDownloader;
|
module.exports = FileDownloader;
|
||||||
|
|||||||
@@ -49,7 +49,7 @@ class BaseLog {
|
|||||||
this.outputBuffer = [];
|
this.outputBuffer = [];
|
||||||
|
|
||||||
await this.flushImpl(this.data)
|
await this.flushImpl(this.data)
|
||||||
.catch(e => { console.log(e); ayncExit.exit(1); } );
|
.catch(e => { console.error(`Logger error: ${e}`); ayncExit.exit(1); } );
|
||||||
this.flushing = false;
|
this.flushing = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -188,8 +188,8 @@ class Logger {
|
|||||||
}
|
}
|
||||||
|
|
||||||
this.closed = false;
|
this.closed = false;
|
||||||
ayncExit.onSignal((signal) => {
|
ayncExit.onSignal((signal, err) => {
|
||||||
this.log(LM_FATAL, `Signal ${signal} received, exiting...`);
|
this.log(LM_FATAL, `Signal "${signal}" received, error: "${(err.stack ? err.stack : err)}", exiting...`);
|
||||||
});
|
});
|
||||||
ayncExit.addAfter(this.close.bind(this));
|
ayncExit.addAfter(this.close.bind(this));
|
||||||
}
|
}
|
||||||
@@ -218,6 +218,8 @@ class Logger {
|
|||||||
} else {
|
} else {
|
||||||
console.log(mes);
|
console.log(mes);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return mes;
|
||||||
}
|
}
|
||||||
|
|
||||||
async close() {
|
async close() {
|
||||||
|
|||||||
@@ -3,7 +3,7 @@ const chardet = require('chardet');
|
|||||||
function getEncoding(buf) {
|
function getEncoding(buf) {
|
||||||
let selected = getEncodingLite(buf);
|
let selected = getEncodingLite(buf);
|
||||||
|
|
||||||
if (selected == 'ISO-8859-5') {
|
if (selected == 'ISO-8859-5' && buf.length > 10) {
|
||||||
const charsetAll = chardet.analyse(buf.slice(0, 20000));
|
const charsetAll = chardet.analyse(buf.slice(0, 20000));
|
||||||
for (const charset of charsetAll) {
|
for (const charset of charsetAll) {
|
||||||
if (charset.name.indexOf('ISO-8859') < 0) {
|
if (charset.name.indexOf('ISO-8859') < 0) {
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ const _ = require('lodash');
|
|||||||
|
|
||||||
const utils = require('../utils');
|
const utils = require('../utils');
|
||||||
const JembaConnManager = require('../../db/JembaConnManager');//singleton
|
const JembaConnManager = require('../../db/JembaConnManager');//singleton
|
||||||
|
const log = new (require('../AppLogger'))().log;//singleton
|
||||||
|
|
||||||
let instance = null;
|
let instance = null;
|
||||||
|
|
||||||
@@ -20,25 +21,30 @@ class JembaReaderStorage {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async doAction(act) {
|
async doAction(act) {
|
||||||
if (!_.isObject(act.items))
|
try {
|
||||||
throw new Error('items is not an object');
|
if (!_.isObject(act.items))
|
||||||
|
throw new Error('items is not an object');
|
||||||
|
|
||||||
let result = {};
|
let result = {};
|
||||||
switch (act.action) {
|
switch (act.action) {
|
||||||
case 'check':
|
case 'check':
|
||||||
result = await this.checkItems(act.items);
|
result = await this.checkItems(act.items);
|
||||||
break;
|
break;
|
||||||
case 'get':
|
case 'get':
|
||||||
result = await this.getItems(act.items);
|
result = await this.getItems(act.items);
|
||||||
break;
|
break;
|
||||||
case 'set':
|
case 'set':
|
||||||
result = await this.setItems(act.items, act.force);
|
result = await this.setItems(act.items, act.force);
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
throw new Error('Unknown action');
|
throw new Error('Unknown action');
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
} catch (e) {
|
||||||
|
log(LM_ERR, `JembaReaderStorage: ${e.message}`);
|
||||||
|
throw e;
|
||||||
}
|
}
|
||||||
|
|
||||||
return result;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
async checkItems(items) {
|
async checkItems(items) {
|
||||||
|
|||||||
@@ -219,6 +219,27 @@ class ReaderWorker {
|
|||||||
return `disk://${hash}`;
|
return `disk://${hash}`;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
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);
|
||||||
|
} 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) {
|
async restoreRemoteFile(filename) {
|
||||||
const basename = path.basename(filename);
|
const basename = path.basename(filename);
|
||||||
const targetName = `${this.config.tempPublicDir}/${basename}`;
|
const targetName = `${this.config.tempPublicDir}/${basename}`;
|
||||||
|
|||||||
@@ -9,8 +9,7 @@ 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) {
|
||||||
//const ws = 'ws';//for nodejs
|
this.WebSocket = (isBrowser ? WebSocket : require('ws'));
|
||||||
this.WebSocket = (isBrowser ? WebSocket : null/*for nodejs require(ws)*/);
|
|
||||||
this.url = url;
|
this.url = url;
|
||||||
this.ws = null;
|
this.ws = null;
|
||||||
this.listeners = [];
|
this.listeners = [];
|
||||||
@@ -95,7 +94,7 @@ class WebSocketConnection {
|
|||||||
this.ws = new this.WebSocket(this.url);
|
this.ws = new this.WebSocket(this.url);
|
||||||
}
|
}
|
||||||
|
|
||||||
const onopen = (e) => {
|
const onopen = () => {
|
||||||
this.connecting = false;
|
this.connecting = false;
|
||||||
resolve(this.ws);
|
resolve(this.ws);
|
||||||
};
|
};
|
||||||
@@ -166,7 +165,7 @@ class WebSocketConnection {
|
|||||||
this.requestId = (this.requestId < 1000000 ? this.requestId + 1 : 1);
|
this.requestId = (this.requestId < 1000000 ? this.requestId + 1 : 1);
|
||||||
const requestId = this.requestId;//реентерабельность!!!
|
const requestId = this.requestId;//реентерабельность!!!
|
||||||
|
|
||||||
this.ws.send(JSON.stringify(Object.assign({requestId, _rpo: 1}, req)));//_rpo: 1 - ждем в ответ _rok: 1
|
this.ws.send(JSON.stringify(Object.assign({requestId}, req)));
|
||||||
|
|
||||||
let resp = {};
|
let resp = {};
|
||||||
try {
|
try {
|
||||||
|
|||||||
@@ -13,7 +13,7 @@ function toBase36(data) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function fromBase36(data) {
|
function fromBase36(data) {
|
||||||
return bs36.decode(data);
|
return Buffer.from(bs36.decode(data));
|
||||||
}
|
}
|
||||||
|
|
||||||
function bufferRemoveZeroes(buf) {
|
function bufferRemoveZeroes(buf) {
|
||||||
@@ -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,
|
||||||
|
|||||||
@@ -2,7 +2,7 @@ const fs = require('fs-extra');
|
|||||||
const _ = require('lodash');
|
const _ = require('lodash');
|
||||||
|
|
||||||
const ayncExit = new (require('../core/AsyncExit'))();//singleton
|
const ayncExit = new (require('../core/AsyncExit'))();//singleton
|
||||||
const { JembaDb, JembaDbThread } = require('./JembaDb');
|
const { JembaDb, JembaDbThread } = require('jembadb');
|
||||||
const log = new (require('../core/AppLogger'))().log;//singleton
|
const log = new (require('../core/AppLogger'))().log;//singleton
|
||||||
|
|
||||||
const jembaMigrations = require('./jembaMigrations');
|
const jembaMigrations = require('./jembaMigrations');
|
||||||
@@ -14,6 +14,7 @@ class JembaConnManager {
|
|||||||
constructor() {
|
constructor() {
|
||||||
if (!instance) {
|
if (!instance) {
|
||||||
this.inited = false;
|
this.inited = false;
|
||||||
|
this._db = {};
|
||||||
|
|
||||||
instance = this;
|
instance = this;
|
||||||
}
|
}
|
||||||
@@ -28,6 +29,8 @@ class JembaConnManager {
|
|||||||
this.config = config;
|
this.config = config;
|
||||||
this._db = {};
|
this._db = {};
|
||||||
|
|
||||||
|
ayncExit.add(this.close.bind(this));
|
||||||
|
|
||||||
for (const dbConfig of this.config.jembaDb) {
|
for (const dbConfig of this.config.jembaDb) {
|
||||||
const dbPath = `${this.config.dataDir}/db/${dbConfig.dbName}`;
|
const dbPath = `${this.config.dataDir}/db/${dbConfig.dbName}`;
|
||||||
|
|
||||||
@@ -44,11 +47,23 @@ class JembaConnManager {
|
|||||||
} else {
|
} else {
|
||||||
dbConn = new JembaDb();
|
dbConn = new JembaDb();
|
||||||
}
|
}
|
||||||
|
this._db[dbConfig.dbName] = dbConn;
|
||||||
|
|
||||||
log(`Open "${dbConfig.dbName}" begin`);
|
log(`Open "${dbConfig.dbName}" begin`);
|
||||||
await dbConn.openDb({dbPath, cacheSize: dbConfig.cacheSize, compressed: dbConfig.compressed, forceFileClosing: dbConfig.forceFileClosing});
|
await dbConn.lock({
|
||||||
|
dbPath,
|
||||||
|
create: true,
|
||||||
|
softLock: true,
|
||||||
|
|
||||||
if (dbConfig.openAll) {
|
tableDefaults: {
|
||||||
|
cacheSize: dbConfig.cacheSize,
|
||||||
|
compressed: dbConfig.compressed,
|
||||||
|
forceFileClosing: dbConfig.forceFileClosing,
|
||||||
|
typeCompatMode: true,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
if (dbConfig.openAll || forceAutoRepair || dbConfig.autoRepair) {
|
||||||
try {
|
try {
|
||||||
await dbConn.openAll();
|
await dbConn.openAll();
|
||||||
} catch(e) {
|
} catch(e) {
|
||||||
@@ -77,21 +92,15 @@ class JembaConnManager {
|
|||||||
if (applied.length)
|
if (applied.length)
|
||||||
log(`${applied.length} migrations applied to "${dbConfig.dbName}"`);
|
log(`${applied.length} migrations applied to "${dbConfig.dbName}"`);
|
||||||
}
|
}
|
||||||
|
|
||||||
this._db[dbConfig.dbName] = dbConn;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ayncExit.add(this.close.bind(this));
|
|
||||||
|
|
||||||
this.inited = true;
|
this.inited = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
async close() {
|
async close() {
|
||||||
if (!this.inited)
|
|
||||||
return;
|
|
||||||
|
|
||||||
for (const dbConfig of this.config.jembaDb) {
|
for (const dbConfig of this.config.jembaDb) {
|
||||||
await this._db[dbConfig.dbName].closeDb();
|
if (this._db[dbConfig.dbName])
|
||||||
|
await this._db[dbConfig.dbName].unlock();
|
||||||
}
|
}
|
||||||
|
|
||||||
this._db = {};
|
this._db = {};
|
||||||
|
|||||||
@@ -1,536 +0,0 @@
|
|||||||
const fs = require('fs').promises;
|
|
||||||
|
|
||||||
const Table = require('./Table');
|
|
||||||
const utils = require('./utils');
|
|
||||||
|
|
||||||
/* API methods:
|
|
||||||
openDb
|
|
||||||
closeDb
|
|
||||||
|
|
||||||
create
|
|
||||||
drop
|
|
||||||
|
|
||||||
open
|
|
||||||
openAll
|
|
||||||
close
|
|
||||||
closeAll
|
|
||||||
|
|
||||||
tableExists
|
|
||||||
getDbInfo
|
|
||||||
getDbSize
|
|
||||||
|
|
||||||
select
|
|
||||||
insert
|
|
||||||
update
|
|
||||||
delete
|
|
||||||
|
|
||||||
esc
|
|
||||||
*/
|
|
||||||
|
|
||||||
class JembaDb {
|
|
||||||
constructor() {
|
|
||||||
this.opened = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
query = {
|
|
||||||
dbPath: String,
|
|
||||||
//table open defaults
|
|
||||||
inMemory: Boolean, false
|
|
||||||
cacheSize: Number, 5
|
|
||||||
compressed: Number, {0..9}, 0
|
|
||||||
recreate: Boolean, false,
|
|
||||||
autoRepair: Boolean, false,
|
|
||||||
forceFileClosing: Boolean, false,
|
|
||||||
lazyOpen: Boolean, false,
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
async openDb(query = {}) {
|
|
||||||
if (this.opened)
|
|
||||||
throw new Error(`Database ${this.dbPath} has already been opened`);
|
|
||||||
|
|
||||||
if (!query.dbPath)
|
|
||||||
throw new Error(`'query.dbPath' parameter is required`);
|
|
||||||
|
|
||||||
this.dbPath = query.dbPath;
|
|
||||||
await fs.mkdir(this.dbPath, { recursive: true });
|
|
||||||
|
|
||||||
this.table = new Map();
|
|
||||||
this.tableOpenDefaults = {
|
|
||||||
inMemory: query.inMemory,
|
|
||||||
cacheSize: query.cacheSize,
|
|
||||||
compressed: query.compressed,
|
|
||||||
recreate: query.recreate,
|
|
||||||
autoRepair: query.autoRepair,
|
|
||||||
forceFileClosing: query.forceFileClosing,
|
|
||||||
lazyOpen: query.lazyOpen,
|
|
||||||
};
|
|
||||||
|
|
||||||
this.opened = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
async closeDb() {
|
|
||||||
if (!this.opened)
|
|
||||||
return;
|
|
||||||
|
|
||||||
await this.closeAll();
|
|
||||||
this.opened = false;
|
|
||||||
|
|
||||||
//console.log('closed');
|
|
||||||
}
|
|
||||||
|
|
||||||
checkOpened() {
|
|
||||||
if (!this.opened)
|
|
||||||
throw new Error('Database closed');
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
query = {
|
|
||||||
table: 'tableName',
|
|
||||||
quietIfExists: Boolean,
|
|
||||||
inMemory: Boolean, false
|
|
||||||
cacheSize: Number, 5
|
|
||||||
compressed: Number, {0..9}, 0
|
|
||||||
recreate: Boolean, false,
|
|
||||||
autoRepair: Boolean, false,
|
|
||||||
forceFileClosing: Boolean, false,
|
|
||||||
lazyOpen: Boolean, false,
|
|
||||||
|
|
||||||
in: 'tableName',
|
|
||||||
flag: Object || Array, {name: 'flag1', check: '(r) => r.id > 10'}
|
|
||||||
hash: Object || Array, {field: 'field1', type: 'string', depth: 11, allowUndef: false}
|
|
||||||
index: Object || Array, {field: 'field1', type: 'string', depth: 11, allowUndef: false}
|
|
||||||
}
|
|
||||||
result = {}
|
|
||||||
*/
|
|
||||||
async create(query = {}) {
|
|
||||||
this.checkOpened();
|
|
||||||
|
|
||||||
if ((!query.table && !query.in) || (query.table && query.in))
|
|
||||||
throw new Error(`One of 'query.table' or 'query.in' parameters is required, but not both`);
|
|
||||||
|
|
||||||
let table;
|
|
||||||
if (query.table) {
|
|
||||||
if (await this.tableExists({table: query.table})) {
|
|
||||||
if (!query.quietIfExists)
|
|
||||||
throw new Error(`Table '${query.table}' already exists`);
|
|
||||||
|
|
||||||
table = this.table.get(query.table);
|
|
||||||
} else {
|
|
||||||
table = new Table();
|
|
||||||
this.table.set(query.table, table);
|
|
||||||
|
|
||||||
await this.open(query);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if (await this.tableExists({table: query.in})) {
|
|
||||||
table = this.table.get(query.in);
|
|
||||||
} else {
|
|
||||||
throw new Error(`Table '${query.in}' does not exist`);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (query.flag || query.hash || query.index) {
|
|
||||||
await table.create({
|
|
||||||
quietIfExists: query.quietIfExists,
|
|
||||||
flag: query.flag,
|
|
||||||
hash: query.hash,
|
|
||||||
index: query.index,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
return {};
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
query = {
|
|
||||||
table: 'tableName',
|
|
||||||
|
|
||||||
in: 'tableName',
|
|
||||||
flag: Object || Array, {name: 'flag1'}
|
|
||||||
hash: Object || Array, {field: 'field1'}
|
|
||||||
index: Object || Array, {field: 'field1'}
|
|
||||||
}
|
|
||||||
result = {}
|
|
||||||
*/
|
|
||||||
async drop(query = {}) {
|
|
||||||
this.checkOpened();
|
|
||||||
|
|
||||||
if ((!query.table && !query.in) || (query.table && query.in))
|
|
||||||
throw new Error(`One of 'query.table' or 'query.in' parameters is required, but not both`);
|
|
||||||
|
|
||||||
if (query.table) {
|
|
||||||
if (await this.tableExists({table: query.table})) {
|
|
||||||
const table = this.table.get(query.table);
|
|
||||||
if (table && table.opened) {
|
|
||||||
await table.close();
|
|
||||||
}
|
|
||||||
|
|
||||||
const basePath = `${this.dbPath}/${query.table}`;
|
|
||||||
await fs.rmdir(basePath, { recursive: true });
|
|
||||||
|
|
||||||
this.table.delete(query.table);
|
|
||||||
} else {
|
|
||||||
throw new Error(`Table '${query.table}' does not exist`);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if (await this.tableExists({table: query.in})) {
|
|
||||||
const table = this.table.get(query.in);
|
|
||||||
|
|
||||||
if (table) {
|
|
||||||
if (query.flag || query.hash || query.index) {
|
|
||||||
await table.drop({
|
|
||||||
flag: query.flag,
|
|
||||||
hash: query.hash,
|
|
||||||
index: query.index,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
throw new Error(`Table '${query.in}' has not been opened yet`);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
throw new Error(`Table '${query.in}' does not exist`);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return {};
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
query = {
|
|
||||||
(!) table: 'tableName',
|
|
||||||
inMemory: Boolean, false
|
|
||||||
cacheSize: Number, 5
|
|
||||||
compressed: Number, {0..9}, 0
|
|
||||||
recreate: Boolean, false,
|
|
||||||
autoRepair: Boolean, false,
|
|
||||||
forceFileClosing: Boolean, false,
|
|
||||||
lazyOpen: Boolean, false,
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
async open(query = {}) {
|
|
||||||
this.checkOpened();
|
|
||||||
|
|
||||||
if (!query.table)
|
|
||||||
throw new Error(`'query.table' parameter is required`);
|
|
||||||
|
|
||||||
if (await this.tableExists({table: query.table})) {
|
|
||||||
let table = this.table.get(query.table);
|
|
||||||
|
|
||||||
if (!table) {
|
|
||||||
table = new Table();
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!table.opened) {
|
|
||||||
const opts = Object.assign({}, this.tableOpenDefaults, query);
|
|
||||||
opts.tablePath = `${this.dbPath}/${query.table}`;
|
|
||||||
await table.open(opts);
|
|
||||||
}
|
|
||||||
|
|
||||||
this.table.set(query.table, table);
|
|
||||||
} else {
|
|
||||||
throw new Error(`Table '${query.table}' does not exist`);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
async _getTableList() {
|
|
||||||
const result = [];
|
|
||||||
const files = await fs.readdir(this.dbPath, { withFileTypes: true });
|
|
||||||
|
|
||||||
for (const file of files) {
|
|
||||||
if (file.isDirectory()) {
|
|
||||||
if (file.name.indexOf('___temporary_recreating') >= 0)
|
|
||||||
continue;
|
|
||||||
result.push(file.name);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
query = {
|
|
||||||
inMemory: Boolean, false
|
|
||||||
cacheSize: Number, 5
|
|
||||||
compressed: Number, {0..9}, 0
|
|
||||||
recreate: Boolean, false,
|
|
||||||
autoRepair: Boolean, false,
|
|
||||||
forceFileClosing: Boolean, false,
|
|
||||||
lazyOpen: Boolean, false,
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
async openAll(query = {}) {
|
|
||||||
this.checkOpened();
|
|
||||||
|
|
||||||
const tables = await this._getTableList();
|
|
||||||
|
|
||||||
//sequentially
|
|
||||||
for (const table of tables) {
|
|
||||||
this.checkOpened();
|
|
||||||
await this.open(Object.assign({}, query, {table}));
|
|
||||||
}
|
|
||||||
|
|
||||||
/*const promises = [];
|
|
||||||
for (const table of tables) {
|
|
||||||
promises.push(this.open(Object.assign({}, query, {table})));
|
|
||||||
}
|
|
||||||
await Promise.all(promises);*/
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
query = {
|
|
||||||
(!) table: 'tableName',
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
async close(query = {}) {
|
|
||||||
this.checkOpened();
|
|
||||||
|
|
||||||
if (!query.table)
|
|
||||||
throw new Error(`'query.table' parameter is required`);
|
|
||||||
|
|
||||||
if (await this.tableExists({table: query.table})) {
|
|
||||||
let table = this.table.get(query.table);
|
|
||||||
|
|
||||||
if (table) {
|
|
||||||
await table.close();
|
|
||||||
}
|
|
||||||
|
|
||||||
this.table.delete(query.table);
|
|
||||||
} else {
|
|
||||||
throw new Error(`Table '${query.table}' does not exist`);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async closeAll() {
|
|
||||||
this.checkOpened();
|
|
||||||
|
|
||||||
const promises = [];
|
|
||||||
for (const table of this.table.keys()) {
|
|
||||||
promises.push(this.close({table}));
|
|
||||||
}
|
|
||||||
await Promise.all(promises);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
query = {
|
|
||||||
(!) table: 'tableName'
|
|
||||||
},
|
|
||||||
result = Boolean
|
|
||||||
*/
|
|
||||||
async tableExists(query = {}) {
|
|
||||||
this.checkOpened();
|
|
||||||
|
|
||||||
if (!query.table)
|
|
||||||
throw new Error(`'query.table' parameter is required`);
|
|
||||||
|
|
||||||
if (this.table.has(query.table))
|
|
||||||
return true;
|
|
||||||
|
|
||||||
if (await utils.pathExists(`${this.dbPath}/${query.table}`))
|
|
||||||
return true;
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
query = {
|
|
||||||
table: 'tableName'
|
|
||||||
},
|
|
||||||
result = {
|
|
||||||
dbPath: String,
|
|
||||||
tableName1: {opened: Boolean, ...},
|
|
||||||
tableName2: {opened: Boolean, ...},
|
|
||||||
...
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
async getDbInfo(query = {}) {
|
|
||||||
this.checkOpened();
|
|
||||||
|
|
||||||
const tables = await this._getTableList();
|
|
||||||
|
|
||||||
const result = {dbPath: this.dbPath};
|
|
||||||
for (const table of tables) {
|
|
||||||
if (!query.table || (query.table && table == query.table)) {
|
|
||||||
const tableInstance = this.table.get(table);
|
|
||||||
if (tableInstance && tableInstance.opened) {
|
|
||||||
result[table] = await tableInstance.getMeta();
|
|
||||||
result[table].opened = true;
|
|
||||||
} else {
|
|
||||||
result[table] = {opened: false};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
result = {
|
|
||||||
total: Number,
|
|
||||||
tables: {
|
|
||||||
tableName1: Number,
|
|
||||||
tableName2: Number,
|
|
||||||
...
|
|
||||||
}
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
async getDbSize() {
|
|
||||||
this.checkOpened();
|
|
||||||
|
|
||||||
const dirs = await fs.readdir(this.dbPath, { withFileTypes: true });
|
|
||||||
|
|
||||||
const result = {total: 0, tables: {}};
|
|
||||||
for (const dir of dirs) {
|
|
||||||
if (dir.isDirectory()) {
|
|
||||||
const table = dir.name;
|
|
||||||
const tablePath = `${this.dbPath}/${table}`;
|
|
||||||
const files = await fs.readdir(tablePath, { withFileTypes: true });
|
|
||||||
|
|
||||||
if (!result.tables[table])
|
|
||||||
result.tables[table] = 0;
|
|
||||||
|
|
||||||
for (const file of files) {
|
|
||||||
if (file.isFile()) {
|
|
||||||
let size = 0;
|
|
||||||
try {
|
|
||||||
size = (await fs.stat(`${tablePath}/${file.name}`)).size;
|
|
||||||
} catch(e) {
|
|
||||||
//
|
|
||||||
}
|
|
||||||
result.tables[table] += size;
|
|
||||||
result.total += size;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
query = {
|
|
||||||
(!) table: 'tableName',
|
|
||||||
distinct: 'fieldName' || Array,
|
|
||||||
count: Boolean,
|
|
||||||
map: '(r) => ({id1: r.id, ...})',
|
|
||||||
where: `@@index('field1', 10, 20)`,
|
|
||||||
sort: '(a, b) => a.id - b.id',
|
|
||||||
limit: 10,
|
|
||||||
offset: 10,
|
|
||||||
}
|
|
||||||
result = Array
|
|
||||||
*/
|
|
||||||
async select(query = {}) {
|
|
||||||
this.checkOpened();
|
|
||||||
|
|
||||||
if (!query.table)
|
|
||||||
throw new Error(`'query.table' parameter is required`);
|
|
||||||
|
|
||||||
const table = this.table.get(query.table);
|
|
||||||
if (table) {
|
|
||||||
return await table.select(query);
|
|
||||||
} else {
|
|
||||||
if (await this.tableExists({table: query.table})) {
|
|
||||||
throw new Error(`Table '${query.table}' has not been opened yet`);
|
|
||||||
} else {
|
|
||||||
throw new Error(`Table '${query.table}' does not exist`);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
query = {
|
|
||||||
(!) table: 'tableName',
|
|
||||||
replace: Boolean,
|
|
||||||
(!) rows: Array,
|
|
||||||
}
|
|
||||||
result = {
|
|
||||||
(!) inserted: Number,
|
|
||||||
(!) replaced: Number,
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
async insert(query = {}) {
|
|
||||||
this.checkOpened();
|
|
||||||
|
|
||||||
if (!query.table)
|
|
||||||
throw new Error(`'query.table' parameter is required`);
|
|
||||||
|
|
||||||
const table = this.table.get(query.table);
|
|
||||||
if (table) {
|
|
||||||
return await table.insert(query);
|
|
||||||
} else {
|
|
||||||
if (await this.tableExists({table: query.table})) {
|
|
||||||
throw new Error(`Table '${query.table}' has not been opened yet`);
|
|
||||||
} else {
|
|
||||||
throw new Error(`Table '${query.table}' does not exist`);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
query = {
|
|
||||||
(!) table: 'tableName',
|
|
||||||
(!) mod: '(r) => r.count++',
|
|
||||||
where: `@@index('field1', 10, 20)`,
|
|
||||||
sort: '(a, b) => a.id - b.id',
|
|
||||||
limit: 10,
|
|
||||||
offset: 10,
|
|
||||||
}
|
|
||||||
result = {
|
|
||||||
(!) updated: Number,
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
async update(query = {}) {
|
|
||||||
this.checkOpened();
|
|
||||||
|
|
||||||
if (!query.table)
|
|
||||||
throw new Error(`'query.table' parameter is required`);
|
|
||||||
|
|
||||||
const table = this.table.get(query.table);
|
|
||||||
if (table) {
|
|
||||||
return await table.update(query);
|
|
||||||
} else {
|
|
||||||
if (await this.tableExists({table: query.table})) {
|
|
||||||
throw new Error(`Table '${query.table}' has not been opened yet`);
|
|
||||||
} else {
|
|
||||||
throw new Error(`Table '${query.table}' does not exist`);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
query = {
|
|
||||||
(!) table: 'tableName',
|
|
||||||
where: `@@index('field1', 10, 20)`,
|
|
||||||
sort: '(a, b) => a.id - b.id',
|
|
||||||
limit: 10,
|
|
||||||
offset: 10,
|
|
||||||
}
|
|
||||||
result = {
|
|
||||||
(!) deleted: Number,
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
async delete(query = {}) {
|
|
||||||
this.checkOpened();
|
|
||||||
|
|
||||||
if (!query.table)
|
|
||||||
throw new Error(`'query.table' parameter is required`);
|
|
||||||
|
|
||||||
const table = this.table.get(query.table);
|
|
||||||
if (table) {
|
|
||||||
return await table.delete(query);
|
|
||||||
} else {
|
|
||||||
if (await this.tableExists({table: query.table})) {
|
|
||||||
throw new Error(`Table '${query.table}' has not been opened yet`);
|
|
||||||
} else {
|
|
||||||
throw new Error(`Table '${query.table}' does not exist`);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
esc(obj) {
|
|
||||||
return utils.esc(obj);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
module.exports = JembaDb;
|
|
||||||
@@ -1,49 +0,0 @@
|
|||||||
const { parentPort } = require('worker_threads');
|
|
||||||
|
|
||||||
const JembaDb = require('./JembaDb');
|
|
||||||
|
|
||||||
const db = new JembaDb();
|
|
||||||
|
|
||||||
if (parentPort) {
|
|
||||||
parentPort.on('message', async(mes) => {
|
|
||||||
let result = {};
|
|
||||||
try {
|
|
||||||
if (db[mes.action])
|
|
||||||
result.result = await db[mes.action](mes.query);
|
|
||||||
else
|
|
||||||
result = {error: 'Action not found: ' + mes.action};
|
|
||||||
} catch (e) {
|
|
||||||
result = {error: e.message};
|
|
||||||
}
|
|
||||||
|
|
||||||
result.requestId = mes.requestId;
|
|
||||||
parentPort.postMessage(result);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
//This is for proper working of pkg (by zeit) and worker_threads
|
|
||||||
//just a copy of the above code as a string
|
|
||||||
module.exports = `
|
|
||||||
const { parentPort } = require('worker_threads');
|
|
||||||
|
|
||||||
const JembaDb = require('./JembaDb');
|
|
||||||
|
|
||||||
const db = new JembaDb();
|
|
||||||
|
|
||||||
if (parentPort) {
|
|
||||||
parentPort.on('message', async(mes) => {
|
|
||||||
let result = {};
|
|
||||||
try {
|
|
||||||
if (db[mes.action])
|
|
||||||
result.result = await db[mes.action](mes.query);
|
|
||||||
else
|
|
||||||
result = {error: 'Action not found: ' + mes.action};
|
|
||||||
} catch (e) {
|
|
||||||
result = {error: e.message};
|
|
||||||
}
|
|
||||||
|
|
||||||
result.requestId = mes.requestId;
|
|
||||||
parentPort.postMessage(result);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
`.replace('./JembaDb', `${__dirname.replace(/\\/g, '/')}/JembaDb`);
|
|
||||||
@@ -1,119 +0,0 @@
|
|||||||
const { Worker } = require('worker_threads');
|
|
||||||
const utils = require('./utils');
|
|
||||||
const JembaDbChild = require('./JembaDbChild');
|
|
||||||
/* API methods:
|
|
||||||
openDb
|
|
||||||
closeDb
|
|
||||||
|
|
||||||
create
|
|
||||||
drop
|
|
||||||
|
|
||||||
open
|
|
||||||
openAll
|
|
||||||
close
|
|
||||||
closeAll
|
|
||||||
|
|
||||||
tableExists
|
|
||||||
getInfo
|
|
||||||
getDbSize
|
|
||||||
|
|
||||||
select
|
|
||||||
insert
|
|
||||||
update
|
|
||||||
delete
|
|
||||||
|
|
||||||
esc
|
|
||||||
*/
|
|
||||||
|
|
||||||
class JembaDbThread {
|
|
||||||
constructor() {
|
|
||||||
this.worker = null;
|
|
||||||
this.listeners = new Map();
|
|
||||||
this.requestId = 0;
|
|
||||||
|
|
||||||
const apiMethods = [
|
|
||||||
'create', 'drop', 'open', 'openAll', 'close', 'closeAll',
|
|
||||||
'tableExists', 'getDbInfo', 'getDbSize', 'select', 'insert', 'update', 'delete', 'dumpTables'
|
|
||||||
];
|
|
||||||
|
|
||||||
for (const action of apiMethods) {
|
|
||||||
this[action] = async(query) => this._action(action, query);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
_terminate() {
|
|
||||||
if (this.worker) {
|
|
||||||
for (const listener of this.listeners.values()) {
|
|
||||||
listener({error: 'Worker terminated'});
|
|
||||||
}
|
|
||||||
this.worker.terminate();
|
|
||||||
}
|
|
||||||
this.worker = null;
|
|
||||||
}
|
|
||||||
|
|
||||||
_runWoker() {
|
|
||||||
//const worker = new Worker(`${__dirname}/JembaDbChild.js`);
|
|
||||||
const worker = new Worker(JembaDbChild, {eval: true});
|
|
||||||
|
|
||||||
worker.on('message', (mes) => {
|
|
||||||
const listener = this.listeners.get(mes.requestId);
|
|
||||||
if (listener)
|
|
||||||
listener(mes);
|
|
||||||
});
|
|
||||||
|
|
||||||
worker.on('error', (err) => {
|
|
||||||
console.error(err);
|
|
||||||
});
|
|
||||||
|
|
||||||
worker.on('exit', () => {
|
|
||||||
this._terminate();
|
|
||||||
});
|
|
||||||
|
|
||||||
this.worker = worker;
|
|
||||||
}
|
|
||||||
|
|
||||||
_action(action, query) {
|
|
||||||
return new Promise((resolve, reject) => {
|
|
||||||
this.requestId++;
|
|
||||||
|
|
||||||
const requestId = this.requestId; //!!!
|
|
||||||
this.listeners.set(requestId, (mes) => {
|
|
||||||
this.listeners.delete(requestId);
|
|
||||||
|
|
||||||
if (mes.error)
|
|
||||||
reject(new Error(mes.error));
|
|
||||||
else
|
|
||||||
resolve(mes.result);
|
|
||||||
});
|
|
||||||
|
|
||||||
if (this.worker) {
|
|
||||||
this.worker.postMessage({requestId: this.requestId, action, query});
|
|
||||||
} else {
|
|
||||||
reject(new Error('Worker does not exist (database closed?)'));
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
async openDb(query = {}) {
|
|
||||||
if (!this.worker) {
|
|
||||||
this._runWoker();
|
|
||||||
} else {
|
|
||||||
throw new Error('Worker has been created already');
|
|
||||||
}
|
|
||||||
|
|
||||||
return this._action('openDb', query);
|
|
||||||
}
|
|
||||||
|
|
||||||
async closeDb() {
|
|
||||||
const result = await this._action('closeDb');
|
|
||||||
this._terminate();
|
|
||||||
//console.log('DB closed');
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
esc(obj) {
|
|
||||||
return utils.esc(obj);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
module.exports = JembaDbThread;
|
|
||||||
@@ -1,38 +0,0 @@
|
|||||||
class LockQueue {
|
|
||||||
constructor(queueSize) {
|
|
||||||
this.queueSize = queueSize;
|
|
||||||
this.freed = true;
|
|
||||||
this.waitingQueue = [];
|
|
||||||
}
|
|
||||||
|
|
||||||
ret() {
|
|
||||||
this.freed = true;
|
|
||||||
if (this.waitingQueue.length) {
|
|
||||||
this.waitingQueue.shift().onFreed();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
get(take = true) {
|
|
||||||
return new Promise((resolve) => {
|
|
||||||
if (this.freed) {
|
|
||||||
if (take)
|
|
||||||
this.freed = false;
|
|
||||||
resolve();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (this.waitingQueue.length >= this.queueSize)
|
|
||||||
throw new Error('Lock queue is too long');
|
|
||||||
|
|
||||||
this.waitingQueue.push({
|
|
||||||
onFreed: () => {
|
|
||||||
if (take)
|
|
||||||
this.freed = false;
|
|
||||||
resolve();
|
|
||||||
},
|
|
||||||
});
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
module.exports = LockQueue;
|
|
||||||
@@ -1,852 +0,0 @@
|
|||||||
const fs = require('fs').promises;
|
|
||||||
const utils = require('./utils');
|
|
||||||
|
|
||||||
const TableReducer = require('./TableReducer');
|
|
||||||
const TableRowsMem = require('./TableRowsMem');
|
|
||||||
const TableRowsFile = require('./TableRowsFile');
|
|
||||||
const LockQueue = require('./LockQueue');
|
|
||||||
|
|
||||||
const maxChangesLength = 10;
|
|
||||||
|
|
||||||
class Table {
|
|
||||||
constructor() {
|
|
||||||
this.rowsInterface = new TableRowsMem();
|
|
||||||
|
|
||||||
this.autoIncrement = 0;
|
|
||||||
this.fileError = '';
|
|
||||||
|
|
||||||
this.openingLock = new LockQueue(100);
|
|
||||||
this.lock = new LockQueue(100);
|
|
||||||
|
|
||||||
this.opened = false;
|
|
||||||
this.closed = false;
|
|
||||||
this.deltaStep = 0;
|
|
||||||
this.changes = [];
|
|
||||||
|
|
||||||
//table options defaults
|
|
||||||
this.inMemory = false;
|
|
||||||
this.compressed = 0;
|
|
||||||
this.cacheSize = 5;
|
|
||||||
this.compressed = 0;
|
|
||||||
this.recreate = false;
|
|
||||||
this.autoRepair = false;
|
|
||||||
this.forceFileClosing = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
checkErrors() {
|
|
||||||
if (this.fileError)
|
|
||||||
throw new Error(this.fileError);
|
|
||||||
|
|
||||||
if (this.closed)
|
|
||||||
throw new Error('Table closed');
|
|
||||||
|
|
||||||
if (!this.opened)
|
|
||||||
throw new Error('Table has not been opened yet');
|
|
||||||
}
|
|
||||||
|
|
||||||
async waitForSaveChanges() {
|
|
||||||
if (this.changes.length > maxChangesLength) {
|
|
||||||
let i = this.changes.length - maxChangesLength;
|
|
||||||
while (i > 0 && this.changes.length > maxChangesLength) {
|
|
||||||
i--;
|
|
||||||
await utils.sleep(10);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async recreateTable() {
|
|
||||||
const tempTablePath = `${this.tablePath}___temporary_recreating`;
|
|
||||||
await fs.rmdir(tempTablePath, { recursive: true });
|
|
||||||
await fs.mkdir(tempTablePath, { recursive: true });
|
|
||||||
|
|
||||||
const tableRowsFileSrc = new TableRowsFile(this.tablePath, this.cacheSize);
|
|
||||||
|
|
||||||
const tableRowsFileDest = new TableRowsFile(tempTablePath, this.cacheSize, this.compressed);
|
|
||||||
const reducerDest = new TableReducer(false, tempTablePath, this.compressed, tableRowsFileDest);
|
|
||||||
|
|
||||||
try {
|
|
||||||
await tableRowsFileSrc.loadCorrupted();
|
|
||||||
} catch (e) {
|
|
||||||
console.error(e);
|
|
||||||
}
|
|
||||||
try {
|
|
||||||
await reducerDest._load(true, `${this.tablePath}/meta.0`);
|
|
||||||
} catch (e) {
|
|
||||||
console.error(e);
|
|
||||||
}
|
|
||||||
|
|
||||||
const putRows = async(rows) => {
|
|
||||||
const oldRows = [];
|
|
||||||
const newRows = [];
|
|
||||||
const newRowsStr = [];
|
|
||||||
//checks
|
|
||||||
for (const row of rows) {
|
|
||||||
if (!row) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
const t = typeof(row.id);
|
|
||||||
if (t !== 'number' && t !== 'string') {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
const oldRow = await tableRowsFileDest.getRow(row.id);
|
|
||||||
|
|
||||||
if (oldRow) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
let str = '';
|
|
||||||
try {
|
|
||||||
str = JSON.stringify(row);//because of stringify errors
|
|
||||||
} catch(e) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
newRows.push(row);
|
|
||||||
oldRows.push({});
|
|
||||||
newRowsStr.push(str);
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
//reducer
|
|
||||||
reducerDest._update(oldRows, newRows, 1);
|
|
||||||
|
|
||||||
//insert
|
|
||||||
for (let i = 0; i < newRows.length; i++) {
|
|
||||||
const newRow = newRows[i];
|
|
||||||
const newRowStr = newRowsStr[i];
|
|
||||||
|
|
||||||
tableRowsFileDest.setRow(newRow.id, newRow, newRowStr, 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
await tableRowsFileDest.saveDelta(1);
|
|
||||||
await reducerDest._saveDelta(1);
|
|
||||||
} catch(e) {
|
|
||||||
console.error(e);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
let rows = [];
|
|
||||||
for (const id of tableRowsFileSrc.getAllIds()) {
|
|
||||||
if (this.closed)
|
|
||||||
throw new Error('Table closed');
|
|
||||||
|
|
||||||
let row = null;
|
|
||||||
try {
|
|
||||||
row = await tableRowsFileSrc.getRow(id);
|
|
||||||
} catch(e) {
|
|
||||||
console.error(e);
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
rows.push(row);
|
|
||||||
if (rows.length > 1000) {
|
|
||||||
await putRows(rows);
|
|
||||||
rows = [];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (rows.length)
|
|
||||||
await putRows(rows);
|
|
||||||
|
|
||||||
await tableRowsFileDest.saveDelta(0);
|
|
||||||
|
|
||||||
const delta = reducerDest._getDelta(0);
|
|
||||||
delta.dumpMeta = true;
|
|
||||||
await reducerDest._saveDelta(0);
|
|
||||||
|
|
||||||
await tableRowsFileSrc.destroy();
|
|
||||||
await reducerDest._destroy();
|
|
||||||
await tableRowsFileDest.destroy();
|
|
||||||
|
|
||||||
await fs.writeFile(`${tempTablePath}/state`, '1');
|
|
||||||
|
|
||||||
await fs.rmdir(this.tablePath, { recursive: true });
|
|
||||||
await fs.rename(tempTablePath, this.tablePath);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
query: {
|
|
||||||
tablePath: String,
|
|
||||||
inMemory: Boolean,
|
|
||||||
cacheSize: Number,
|
|
||||||
compressed: Number, 0..9
|
|
||||||
recreate: Boolean, false,
|
|
||||||
autoRepair: Boolean, false,
|
|
||||||
forceFileClosing: Boolean, false,
|
|
||||||
lazyOpen: Boolean, false,
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
async _open(query = {}) {
|
|
||||||
if (this.opening)
|
|
||||||
return;
|
|
||||||
this.opening = true;
|
|
||||||
await this.openingLock.get();
|
|
||||||
//console.log(query);
|
|
||||||
try {
|
|
||||||
if (this.opened)
|
|
||||||
throw new Error('Table has already been opened');
|
|
||||||
if (this.closed)
|
|
||||||
throw new Error('Table instance has been destroyed. Please create a new one.');
|
|
||||||
|
|
||||||
this.inMemory = !!query.inMemory;
|
|
||||||
|
|
||||||
if (this.inMemory) {
|
|
||||||
this.reducer = new TableReducer(this.inMemory, '', 0, this.rowsInterface);
|
|
||||||
} else {
|
|
||||||
if (!query.tablePath)
|
|
||||||
throw new Error(`'query.tablePath' parameter is required`);
|
|
||||||
|
|
||||||
this.tablePath = query.tablePath;
|
|
||||||
this.cacheSize = query.cacheSize || 5;
|
|
||||||
this.compressed = query.compressed || 0;
|
|
||||||
this.recreate = query.recreate || false;
|
|
||||||
this.autoRepair = query.autoRepair || false;
|
|
||||||
this.forceFileClosing = query.forceFileClosing || false;
|
|
||||||
|
|
||||||
await fs.mkdir(this.tablePath, { recursive: true });
|
|
||||||
|
|
||||||
this.tableRowsFile = new TableRowsFile(query.tablePath, this.cacheSize, this.compressed);
|
|
||||||
this.rowsInterface = this.tableRowsFile;
|
|
||||||
|
|
||||||
this.reducer = new TableReducer(this.inMemory, this.tablePath, this.compressed, this.rowsInterface);
|
|
||||||
|
|
||||||
const statePath = `${this.tablePath}/state`;
|
|
||||||
let state = null;
|
|
||||||
if (await utils.pathExists(statePath)) {
|
|
||||||
state = await fs.readFile(statePath, 'utf8');
|
|
||||||
}
|
|
||||||
|
|
||||||
if (state === null) {//check if other files exists
|
|
||||||
const files = await fs.readdir(this.tablePath);
|
|
||||||
if (files.length)
|
|
||||||
state = '0';
|
|
||||||
}
|
|
||||||
|
|
||||||
if (this.recreate) {
|
|
||||||
await this.recreateTable();
|
|
||||||
state = '1';
|
|
||||||
}
|
|
||||||
|
|
||||||
if (state !== null) {
|
|
||||||
try {
|
|
||||||
if (state === '1') {
|
|
||||||
// load tableRowsFile & reducer
|
|
||||||
this.autoIncrement = await this.tableRowsFile.load();
|
|
||||||
await this.reducer._load();
|
|
||||||
} else {
|
|
||||||
throw new Error('Table corrupted')
|
|
||||||
}
|
|
||||||
} catch(e) {
|
|
||||||
if (this.autoRepair) {
|
|
||||||
console.error(e.message);
|
|
||||||
await this.recreateTable();
|
|
||||||
} else {
|
|
||||||
throw e;
|
|
||||||
}
|
|
||||||
// load tableRowsFile & reducer
|
|
||||||
this.autoIncrement = await this.tableRowsFile.load();
|
|
||||||
await this.reducer._load();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
this.opened = true;
|
|
||||||
} catch(e) {
|
|
||||||
await this.close();
|
|
||||||
const errMes = `Open table (${query.tablePath}): ${e.message}`;
|
|
||||||
if (!query.lazyOpen)
|
|
||||||
throw new Error(errMes);
|
|
||||||
else
|
|
||||||
this.fileError = errMes;
|
|
||||||
} finally {
|
|
||||||
this.openingLock.ret();
|
|
||||||
this.opening = false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async open(query = {}) {
|
|
||||||
if (query.lazyOpen) {
|
|
||||||
this._open(query);
|
|
||||||
} else {
|
|
||||||
await this._open(query);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async close() {
|
|
||||||
if (this.closed)
|
|
||||||
return;
|
|
||||||
|
|
||||||
this.opened = false;
|
|
||||||
this.closed = true;
|
|
||||||
|
|
||||||
if (!this.inMemory) {
|
|
||||||
while (this.savingChanges) {
|
|
||||||
await utils.sleep(10);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
//for GC
|
|
||||||
if (this.reducer)
|
|
||||||
await this.reducer._destroy();
|
|
||||||
this.reducer = null;
|
|
||||||
|
|
||||||
if (this.rowsInterface)
|
|
||||||
await this.rowsInterface.destroy();
|
|
||||||
this.rowsInterface = null;
|
|
||||||
this.tableRowsFile = null;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
query = {
|
|
||||||
quietIfExists: Boolean,
|
|
||||||
flag: Object || Array, {name: 'flag1', check: '(r) => r.id > 10'}
|
|
||||||
hash: Object || Array, {field: 'field1', type: 'string', depth: 11, allowUndef: false}
|
|
||||||
index: Object || Array, {field: 'field1', type: 'string', depth: 11, allowUndef: false}
|
|
||||||
}
|
|
||||||
result = {}
|
|
||||||
*/
|
|
||||||
async create(query) {
|
|
||||||
await this.openingLock.get(false);
|
|
||||||
this.checkErrors();
|
|
||||||
|
|
||||||
await this.lock.get();
|
|
||||||
try {
|
|
||||||
this.deltaStep++;
|
|
||||||
try {
|
|
||||||
if (query.flag) {
|
|
||||||
for (const flag of utils.paramToArray(query.flag)) {
|
|
||||||
await this.reducer._addFlag(flag, query.quietIfExists, this.deltaStep);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (query.hash) {
|
|
||||||
for (const hash of utils.paramToArray(query.hash)) {
|
|
||||||
await this.reducer._addHash(hash, query.quietIfExists, this.deltaStep);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (query.index) {
|
|
||||||
for (const index of utils.paramToArray(query.index)) {
|
|
||||||
await this.reducer._addIndex(index, query.quietIfExists, this.deltaStep);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
this.changes.push([this.deltaStep, 1]);
|
|
||||||
} catch(e) {
|
|
||||||
this.changes.push([this.deltaStep, 0]);
|
|
||||||
throw e;
|
|
||||||
}
|
|
||||||
|
|
||||||
return {};
|
|
||||||
} finally {
|
|
||||||
this.saveChanges();//no await
|
|
||||||
this.lock.ret();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
query = {
|
|
||||||
flag: Object || Array, {name: 'flag1'}
|
|
||||||
hash: Object || Array, {field: 'field1'}
|
|
||||||
index: Object || Array, {field: 'field1'}
|
|
||||||
}
|
|
||||||
result = {}
|
|
||||||
*/
|
|
||||||
async drop(query) {
|
|
||||||
await this.openingLock.get(false);
|
|
||||||
this.checkErrors();
|
|
||||||
|
|
||||||
await this.lock.get();
|
|
||||||
try {
|
|
||||||
this.deltaStep++;
|
|
||||||
try {
|
|
||||||
if (query.flag) {
|
|
||||||
for (const flag of utils.paramToArray(query.flag)) {
|
|
||||||
await this.reducer._delFlag(flag.name, this.deltaStep);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (query.hash) {
|
|
||||||
for (const hash of utils.paramToArray(query.hash)) {
|
|
||||||
await this.reducer._delHash(hash.field, this.deltaStep);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (query.index) {
|
|
||||||
for (const index of utils.paramToArray(query.index)) {
|
|
||||||
await this.reducer._delIndex(index.field, this.deltaStep);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
this.changes.push([this.deltaStep, 1]);
|
|
||||||
} catch(e) {
|
|
||||||
this.changes.push([this.deltaStep, 0]);
|
|
||||||
throw e;
|
|
||||||
}
|
|
||||||
|
|
||||||
return {};
|
|
||||||
} finally {
|
|
||||||
this.saveChanges();//no await
|
|
||||||
this.lock.ret();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
result = {
|
|
||||||
inMemory: Boolean,
|
|
||||||
flag: Array, [{name: 'flag1', check: '(r) => r.id > 10'}, ...]
|
|
||||||
hash: Array, [{field: 'field1', type: 'string', depth: 11, allowUndef: false}, ...]
|
|
||||||
index: Array, [{field: 'field1', type: 'string', depth: 11, allowUndef: false}, ...]
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
async getMeta() {
|
|
||||||
this.checkErrors();
|
|
||||||
|
|
||||||
return {
|
|
||||||
inMemory: this.inMemory,
|
|
||||||
flag: this.reducer._listFlag(),
|
|
||||||
hash: this.reducer._listHash(),
|
|
||||||
index: this.reducer._listIndex(),
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
prepareWhere(where) {
|
|
||||||
if (typeof(where) !== 'string')
|
|
||||||
throw new Error('query.where must be a string');
|
|
||||||
|
|
||||||
return `async(__tr) => {${where.replace(/@@/g, 'return await __tr.').replace(/@/g, 'await __tr.')}}`;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
query = {
|
|
||||||
distinct: 'fieldName' || Array,
|
|
||||||
count: Boolean,
|
|
||||||
map: '(r) => ({id1: r.id, ...})',
|
|
||||||
where: `@@index('field1', 10, 20)`,
|
|
||||||
sort: '(a, b) => a.id - b.id',
|
|
||||||
limit: 10,
|
|
||||||
offset: 10,
|
|
||||||
}
|
|
||||||
result = Array
|
|
||||||
*/
|
|
||||||
async select(query = {}) {
|
|
||||||
await this.openingLock.get(false);
|
|
||||||
this.checkErrors();
|
|
||||||
|
|
||||||
let ids;//iterator
|
|
||||||
if (query.where) {
|
|
||||||
const where = this.prepareWhere(query.where);
|
|
||||||
const whereFunc = new Function(`return ${where}`)();
|
|
||||||
|
|
||||||
ids = await whereFunc(this.reducer);
|
|
||||||
} else {
|
|
||||||
ids = this.rowsInterface.getAllIds();
|
|
||||||
}
|
|
||||||
|
|
||||||
let found = [];
|
|
||||||
|
|
||||||
let distinct = () => true;
|
|
||||||
if (query.distinct) {
|
|
||||||
const distFields = (Array.isArray(query.distinct) ? query.distinct : [query.distinct]);
|
|
||||||
const dist = new Map();
|
|
||||||
distinct = (row) => {
|
|
||||||
let uniq = '';
|
|
||||||
for (const field of distFields) {
|
|
||||||
const value = row[field];
|
|
||||||
uniq += `${(value === undefined ? '___' : '')}${field}:${value}`;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (dist.has(uniq))
|
|
||||||
return false;
|
|
||||||
dist.set(uniq, true);
|
|
||||||
return true;
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!query.where && !query.distinct && query.count) {//some optimization
|
|
||||||
found = [{count: this.rowsInterface.getAllIdsSize()}];
|
|
||||||
} else {//full running
|
|
||||||
for (const id of ids) {
|
|
||||||
const row = await this.rowsInterface.getRow(id);
|
|
||||||
|
|
||||||
if (row && distinct(row)) {
|
|
||||||
found.push(row);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (query.count) {
|
|
||||||
found = [{count: found.length}];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
let result = [];
|
|
||||||
if (query.map) {
|
|
||||||
const mapFunc = new Function(`return ${query.map}`)();
|
|
||||||
|
|
||||||
for (const row of found) {
|
|
||||||
result.push(mapFunc(row));
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
result = found;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (query.sort) {
|
|
||||||
const sortFunc = new Function(`return ${query.sort}`)();
|
|
||||||
result.sort(sortFunc);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (query.hasOwnProperty('limit') || query.hasOwnProperty('offset')) {
|
|
||||||
const offset = query.offset || 0;
|
|
||||||
const limit = (query.hasOwnProperty('limit') ? query.limit : result.length);
|
|
||||||
result = result.slice(offset, offset + limit);
|
|
||||||
}
|
|
||||||
|
|
||||||
return utils.cloneDeep(result);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
query = {
|
|
||||||
replace: Boolean,
|
|
||||||
(!) rows: Array,
|
|
||||||
}
|
|
||||||
result = {
|
|
||||||
(!) inserted: Number,
|
|
||||||
(!) replaced: Number,
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
async insert(query = {}) {
|
|
||||||
await this.openingLock.get(false);
|
|
||||||
this.checkErrors();
|
|
||||||
|
|
||||||
await this.lock.get();
|
|
||||||
try {
|
|
||||||
if (!Array.isArray(query.rows)) {
|
|
||||||
throw new Error('query.rows must be an array');
|
|
||||||
}
|
|
||||||
|
|
||||||
const newRows = utils.cloneDeep(query.rows);
|
|
||||||
const replace = query.replace;
|
|
||||||
|
|
||||||
//autoIncrement correction
|
|
||||||
for (const newRow of newRows) {
|
|
||||||
if (typeof(newRow.id) === 'number' && newRow.id >= this.autoIncrement)
|
|
||||||
this.autoIncrement = newRow.id + 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
const oldRows = [];
|
|
||||||
const newRowsStr = [];
|
|
||||||
//checks
|
|
||||||
for (const newRow of newRows) {
|
|
||||||
if (newRow.hasOwnProperty('___meta'))
|
|
||||||
throw new Error(`Use of field with name '___meta' is forbidden`);
|
|
||||||
|
|
||||||
if (newRow.id === undefined) {
|
|
||||||
newRow.id = this.autoIncrement;
|
|
||||||
this.autoIncrement++;
|
|
||||||
}
|
|
||||||
|
|
||||||
const t = typeof(newRow.id);
|
|
||||||
if (t !== 'number' && t !== 'string') {
|
|
||||||
throw new Error(`Row id bad type, 'number' or 'string' expected, got ${t}`);
|
|
||||||
}
|
|
||||||
|
|
||||||
const oldRow = await this.rowsInterface.getRow(newRow.id);
|
|
||||||
|
|
||||||
if (!replace && oldRow) {
|
|
||||||
throw new Error(`Record id:${newRow.id} already exists`);
|
|
||||||
}
|
|
||||||
|
|
||||||
oldRows.push((oldRow ? oldRow : {}));
|
|
||||||
newRowsStr.push(JSON.stringify(newRow));//because of stringify errors
|
|
||||||
}
|
|
||||||
|
|
||||||
const result = {inserted: 0, replaced: 0};
|
|
||||||
this.deltaStep++;
|
|
||||||
try {
|
|
||||||
//reducer
|
|
||||||
this.reducer._update(oldRows, newRows, this.deltaStep);
|
|
||||||
|
|
||||||
//insert
|
|
||||||
for (let i = 0; i < newRows.length; i++) {
|
|
||||||
const newRow = newRows[i];
|
|
||||||
const newRowStr = newRowsStr[i];
|
|
||||||
const oldRow = oldRows[i];
|
|
||||||
|
|
||||||
this.rowsInterface.setRow(newRow.id, newRow, newRowStr, this.deltaStep);
|
|
||||||
|
|
||||||
if (oldRow.id !== undefined)
|
|
||||||
result.replaced++;
|
|
||||||
else
|
|
||||||
result.inserted++;
|
|
||||||
}
|
|
||||||
|
|
||||||
this.changes.push([this.deltaStep, 1]);
|
|
||||||
} catch(e) {
|
|
||||||
this.changes.push([this.deltaStep, 0]);
|
|
||||||
throw e;
|
|
||||||
}
|
|
||||||
|
|
||||||
await this.waitForSaveChanges();
|
|
||||||
return result;
|
|
||||||
} finally {
|
|
||||||
this.saveChanges();//no await
|
|
||||||
this.lock.ret();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
query = {
|
|
||||||
(!) mod: '(r) => r.count++',
|
|
||||||
where: `@@index('field1', 10, 20)`,
|
|
||||||
sort: '(a, b) => a.id - b.id',
|
|
||||||
limit: 10,
|
|
||||||
offset: 10,
|
|
||||||
}
|
|
||||||
result = {
|
|
||||||
(!) updated: Number,
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
async update(query = {}) {
|
|
||||||
await this.openingLock.get(false);
|
|
||||||
this.checkErrors();
|
|
||||||
|
|
||||||
await this.lock.get();
|
|
||||||
try {
|
|
||||||
if (typeof(query.mod) !== 'string') {
|
|
||||||
throw new Error('query.mod must be a string');
|
|
||||||
}
|
|
||||||
const modFunc = new Function(`return ${query.mod}`)();
|
|
||||||
|
|
||||||
//where
|
|
||||||
let ids;//iterator
|
|
||||||
if (query.where) {
|
|
||||||
const where = this.prepareWhere(query.where);
|
|
||||||
const whereFunc = new Function(`return ${where}`)();
|
|
||||||
|
|
||||||
ids = await whereFunc(this.reducer);
|
|
||||||
} else {
|
|
||||||
ids = this.rowsInterface.getAllIds();
|
|
||||||
}
|
|
||||||
|
|
||||||
//oldRows
|
|
||||||
let oldRows = [];
|
|
||||||
for (const id of ids) {
|
|
||||||
const oldRow = await this.rowsInterface.getRow(id);
|
|
||||||
|
|
||||||
if (oldRow) {
|
|
||||||
oldRows.push(oldRow);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (query.sort) {
|
|
||||||
const sortFunc = new Function(`return ${query.sort}`)();
|
|
||||||
oldRows.sort(sortFunc);
|
|
||||||
}
|
|
||||||
let newRows = utils.cloneDeep(oldRows);
|
|
||||||
|
|
||||||
if (query.hasOwnProperty('limit') || query.hasOwnProperty('offset')) {
|
|
||||||
const offset = query.offset || 0;
|
|
||||||
const limit = (query.hasOwnProperty('limit') ? query.limit : newRows.length);
|
|
||||||
newRows = newRows.slice(offset, offset + limit);
|
|
||||||
oldRows = oldRows.slice(offset, offset + limit);
|
|
||||||
}
|
|
||||||
|
|
||||||
//mod & checks
|
|
||||||
const context = {};
|
|
||||||
const newRowsStr = [];
|
|
||||||
for (const newRow of newRows) {
|
|
||||||
modFunc(newRow, context);
|
|
||||||
|
|
||||||
const t = typeof(newRow.id);
|
|
||||||
if (t !== 'number' && t !== 'string') {
|
|
||||||
throw new Error(`Row id bad type, 'number' or 'string' expected, got ${t}`);
|
|
||||||
}
|
|
||||||
|
|
||||||
//autoIncrement correction
|
|
||||||
if (t === 'number' && newRow.id >= this.autoIncrement)
|
|
||||||
this.autoIncrement = newRow.id + 1;
|
|
||||||
|
|
||||||
if (newRow.hasOwnProperty('___meta'))
|
|
||||||
throw new Error(`Use of field with name '___meta' is forbidden`);
|
|
||||||
|
|
||||||
newRowsStr.push(JSON.stringify(newRow));//because of stringify errors
|
|
||||||
}
|
|
||||||
|
|
||||||
this.deltaStep++;
|
|
||||||
const result = {updated: 0};
|
|
||||||
try {
|
|
||||||
//reducer
|
|
||||||
this.reducer._update(oldRows, newRows, this.deltaStep);
|
|
||||||
|
|
||||||
//replace
|
|
||||||
for (let i = 0; i < newRows.length; i++) {
|
|
||||||
const newRow = newRows[i];
|
|
||||||
const newRowStr = newRowsStr[i];
|
|
||||||
|
|
||||||
// oldRow.id === newRow.id always here, so
|
|
||||||
this.rowsInterface.setRow(newRow.id, newRow, newRowStr, this.deltaStep);
|
|
||||||
|
|
||||||
result.updated++;
|
|
||||||
}
|
|
||||||
|
|
||||||
this.changes.push([this.deltaStep, 1]);
|
|
||||||
} catch(e) {
|
|
||||||
this.changes.push([this.deltaStep, 0]);
|
|
||||||
throw e;
|
|
||||||
}
|
|
||||||
|
|
||||||
await this.waitForSaveChanges();
|
|
||||||
return result;
|
|
||||||
} finally {
|
|
||||||
this.saveChanges();//no await
|
|
||||||
this.lock.ret();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
query = {
|
|
||||||
where: `@@index('field1', 10, 20)`,
|
|
||||||
sort: '(a, b) => a.id - b.id',
|
|
||||||
limit: 10,
|
|
||||||
offset: 10,
|
|
||||||
}
|
|
||||||
result = {
|
|
||||||
(!) deleted: Number,
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
async delete(query = {}) {
|
|
||||||
await this.openingLock.get(false);
|
|
||||||
this.checkErrors();
|
|
||||||
|
|
||||||
await this.lock.get();
|
|
||||||
try {
|
|
||||||
//where
|
|
||||||
let ids;//iterator
|
|
||||||
if (query.where) {
|
|
||||||
const where = this.prepareWhere(query.where);
|
|
||||||
const whereFunc = new Function(`return ${where}`)();
|
|
||||||
|
|
||||||
ids = await whereFunc(this.reducer);
|
|
||||||
} else {
|
|
||||||
ids = this.rowsInterface.getAllIds();
|
|
||||||
}
|
|
||||||
|
|
||||||
//oldRows
|
|
||||||
let oldRows = [];
|
|
||||||
let newRows = [];
|
|
||||||
for (const id of ids) {
|
|
||||||
const oldRow = await this.rowsInterface.getRow(id);
|
|
||||||
|
|
||||||
if (oldRow) {
|
|
||||||
oldRows.push(oldRow);
|
|
||||||
newRows.push({});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (query.sort) {
|
|
||||||
const sortFunc = new Function(`return ${query.sort}`)();
|
|
||||||
oldRows.sort(sortFunc);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (query.hasOwnProperty('limit') || query.hasOwnProperty('offset')) {
|
|
||||||
const offset = query.offset || 0;
|
|
||||||
const limit = (query.hasOwnProperty('limit') ? query.limit : newRows.length);
|
|
||||||
newRows = newRows.slice(offset, offset + limit);
|
|
||||||
oldRows = oldRows.slice(offset, offset + limit);
|
|
||||||
}
|
|
||||||
|
|
||||||
this.deltaStep++;
|
|
||||||
const result = {deleted: 0};
|
|
||||||
try {
|
|
||||||
//reducer
|
|
||||||
this.reducer._update(oldRows, newRows, this.deltaStep);
|
|
||||||
|
|
||||||
//delete
|
|
||||||
for (let i = 0; i < oldRows.length; i++) {
|
|
||||||
const oldRow = oldRows[i];
|
|
||||||
|
|
||||||
this.rowsInterface.deleteRow(oldRow.id, this.deltaStep);
|
|
||||||
|
|
||||||
result.deleted++;
|
|
||||||
}
|
|
||||||
|
|
||||||
this.changes.push([this.deltaStep, 1]);
|
|
||||||
} catch(e) {
|
|
||||||
this.changes.push([this.deltaStep, 0]);
|
|
||||||
throw e;
|
|
||||||
}
|
|
||||||
|
|
||||||
await this.waitForSaveChanges();
|
|
||||||
return result;
|
|
||||||
} finally {
|
|
||||||
this.saveChanges();//no await
|
|
||||||
this.lock.ret();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async saveState(state) {
|
|
||||||
await fs.writeFile(`${this.tablePath}/state`, state);
|
|
||||||
}
|
|
||||||
|
|
||||||
async saveChanges() {
|
|
||||||
this.needSaveChanges = true;
|
|
||||||
if (this.savingChanges)
|
|
||||||
return;
|
|
||||||
|
|
||||||
if (this.inMemory) {
|
|
||||||
this.changes = [];
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
this.checkErrors();
|
|
||||||
} catch(e) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
this.savingChanges = true;
|
|
||||||
try {
|
|
||||||
await utils.sleep(0);
|
|
||||||
|
|
||||||
while (this.needSaveChanges) {
|
|
||||||
this.needSaveChanges = false;
|
|
||||||
|
|
||||||
await this.saveState('0');
|
|
||||||
while (this.changes.length) {
|
|
||||||
|
|
||||||
const len = this.changes.length;
|
|
||||||
let i = 0;
|
|
||||||
while (i < len) {
|
|
||||||
const [deltaStep, isOk] = this.changes[i];
|
|
||||||
i++;
|
|
||||||
|
|
||||||
if (isOk) {
|
|
||||||
await this.tableRowsFile.saveDelta(deltaStep);
|
|
||||||
await this.reducer._saveDelta(deltaStep);
|
|
||||||
} else {
|
|
||||||
await this.tableRowsFile.cancelDelta(deltaStep);
|
|
||||||
await this.reducer._cancelDelta(deltaStep);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
this.changes = this.changes.slice(i);
|
|
||||||
}
|
|
||||||
await this.saveState('1');
|
|
||||||
|
|
||||||
if (this.forceFileClosing) {
|
|
||||||
await this.tableRowsFile.closeAllFiles();
|
|
||||||
await this.reducer._closeAllFiles();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} catch(e) {
|
|
||||||
console.error(e.message);
|
|
||||||
this.fileError = e.message;
|
|
||||||
} finally {
|
|
||||||
this.savingChanges = false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
module.exports = Table;
|
|
||||||
@@ -1,22 +0,0 @@
|
|||||||
class TableFlag {
|
|
||||||
constructor(checkCode) {
|
|
||||||
this.checkCode = checkCode;
|
|
||||||
this.checkFunc = eval(checkCode);
|
|
||||||
|
|
||||||
this.flag = new Set();
|
|
||||||
}
|
|
||||||
|
|
||||||
add(row) {
|
|
||||||
if (this.checkFunc(row)) {
|
|
||||||
this.flag.add(row.id);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
del(row) {
|
|
||||||
this.flag.delete(row.id);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
module.exports = TableFlag;
|
|
||||||
@@ -1,172 +0,0 @@
|
|||||||
class TableHash {
|
|
||||||
//opts.type = 'string' || 'number' || 'number_as_string'
|
|
||||||
constructor(opts = {}) {
|
|
||||||
const type = opts.type || 'string';
|
|
||||||
this.depth = opts.depth || 11;
|
|
||||||
this.allowUndef = opts.allowUndef || false;
|
|
||||||
this.unique = opts.unique || false;
|
|
||||||
|
|
||||||
this.hash = new Map();
|
|
||||||
|
|
||||||
this.isNumber = (type === 'number' || type === 'number_as_string');
|
|
||||||
this.numberAsString = (type === 'number_as_string');
|
|
||||||
this.valueAsString = !this.isNumber || this.numberAsString;
|
|
||||||
}
|
|
||||||
|
|
||||||
checkType(v) {
|
|
||||||
if (typeof(v) != 'number' && this.isNumber)
|
|
||||||
throw new Error(`Hashed value must be a number, got type:${typeof(v)}, value:${v}`);
|
|
||||||
|
|
||||||
if (typeof(v) != 'string' && !this.isNumber)
|
|
||||||
throw new Error(`Hashed value must be a string, got type:${typeof(v)}, value:${v}`);
|
|
||||||
}
|
|
||||||
|
|
||||||
prepareValue(v) {
|
|
||||||
let result = v;
|
|
||||||
if (this.numberAsString) {
|
|
||||||
result = v.toString().padStart(this.depth, '0');
|
|
||||||
}
|
|
||||||
if (this.valueAsString && result.length > this.depth)
|
|
||||||
result = result.substring(0, this.depth);
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
add(value, id) {
|
|
||||||
if (value === undefined && this.allowUndef)
|
|
||||||
return;
|
|
||||||
|
|
||||||
this.checkType(value);
|
|
||||||
|
|
||||||
value = this.prepareValue(value);
|
|
||||||
if (this.hash.has(value)) {
|
|
||||||
if (this.unique) {
|
|
||||||
const id_ = this.hash.get(value);
|
|
||||||
if (id_ !== id) {
|
|
||||||
throw new Error(`Collision for unique hash detected: value:${value}, id1:${id_}, id2:${id}`);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
const ids = this.hash.get(value);
|
|
||||||
ids.add(id);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if (this.unique) {
|
|
||||||
this.hash.set(value, id);
|
|
||||||
} else {
|
|
||||||
const ids = new Set();
|
|
||||||
this.hash.set(value, ids);
|
|
||||||
ids.add(id);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return value;
|
|
||||||
}
|
|
||||||
|
|
||||||
del(value, id) {
|
|
||||||
if (value === undefined && this.allowUndef)
|
|
||||||
return;
|
|
||||||
|
|
||||||
this.checkType(value);
|
|
||||||
|
|
||||||
value = this.prepareValue(value);
|
|
||||||
if (this.hash.has(value)) {
|
|
||||||
if (this.unique) {
|
|
||||||
const id_ = this.hash.get(value);
|
|
||||||
if (id_ === id)
|
|
||||||
this.hash.delete(value);
|
|
||||||
} else {
|
|
||||||
const ids = this.hash.get(value);
|
|
||||||
|
|
||||||
ids.delete(id);
|
|
||||||
|
|
||||||
if (!ids.size) {
|
|
||||||
this.hash.delete(value);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return value;
|
|
||||||
}
|
|
||||||
|
|
||||||
reduce(value) {
|
|
||||||
this.checkType(value);
|
|
||||||
|
|
||||||
value = this.prepareValue(value);
|
|
||||||
let result;
|
|
||||||
if (this.hash.has(value)) {
|
|
||||||
if (this.unique) {
|
|
||||||
result = new Set();
|
|
||||||
result.add(this.hash.get(value));
|
|
||||||
} else {
|
|
||||||
result = this.hash.get(value);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
result = new Set();
|
|
||||||
}
|
|
||||||
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
min() {
|
|
||||||
let result = new Set();
|
|
||||||
|
|
||||||
let min = null;
|
|
||||||
let id = null;
|
|
||||||
for (const value of this.hash.keys()) {
|
|
||||||
if (value < min || min === null) {
|
|
||||||
min = value;
|
|
||||||
id = this.hash.get(min);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (id !== null) {
|
|
||||||
if (this.unique)
|
|
||||||
result.add(id);
|
|
||||||
else
|
|
||||||
result = id;
|
|
||||||
}
|
|
||||||
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
max() {
|
|
||||||
let result = new Set();
|
|
||||||
|
|
||||||
let max = null;
|
|
||||||
let id = null;
|
|
||||||
for (const value of this.hash.keys()) {
|
|
||||||
if (value > max || max === null) {
|
|
||||||
max = value;
|
|
||||||
id = this.hash.get(max);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (id !== null) {
|
|
||||||
if (this.unique)
|
|
||||||
result.add(id);
|
|
||||||
else
|
|
||||||
result = id;
|
|
||||||
}
|
|
||||||
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
iter(checkFunc) {
|
|
||||||
const result = new Set();
|
|
||||||
for (const [value, ids] of this.hash.entries()) {
|
|
||||||
const checkResult = checkFunc(value);
|
|
||||||
if (checkResult === undefined)
|
|
||||||
break;
|
|
||||||
if (checkResult) {
|
|
||||||
if (this.unique) {
|
|
||||||
result.add(ids);
|
|
||||||
} else {
|
|
||||||
for (const id of ids)
|
|
||||||
result.add(id);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
module.exports = TableHash;
|
|
||||||
@@ -1,311 +0,0 @@
|
|||||||
const utils = require('./utils');
|
|
||||||
|
|
||||||
class TableIndex {
|
|
||||||
//opts.type = 'string' || 'number' || 'number_as_string'
|
|
||||||
constructor(opts = {}) {
|
|
||||||
const type = opts.type || 'string';
|
|
||||||
this.depth = opts.depth || 11;
|
|
||||||
this.allowUndef = opts.allowUndef || false;
|
|
||||||
this.unique = opts.unique || false;
|
|
||||||
|
|
||||||
this.hash = new Map();
|
|
||||||
this.sorted = [[]];
|
|
||||||
this.delCount = 0;
|
|
||||||
|
|
||||||
this.isNumber = (type === 'number' || type === 'number_as_string');
|
|
||||||
this.numberAsString = (type === 'number_as_string');
|
|
||||||
this.valueAsString = !this.isNumber || this.numberAsString;
|
|
||||||
|
|
||||||
this.cmp = (a, b) => a.localeCompare(b);
|
|
||||||
if (type === 'number') {
|
|
||||||
this.cmp = (a, b) => a - b;
|
|
||||||
} else if (type === 'number_as_string') {
|
|
||||||
this.cmp = (a, b) => (a < b ? -1 : (a > b ? 1 : 0));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
checkType(v) {
|
|
||||||
if (typeof(v) != 'number' && this.isNumber)
|
|
||||||
throw new Error(`Indexed value must be a number, got type:${typeof(v)}, value:${v}`);
|
|
||||||
|
|
||||||
if (typeof(v) != 'string' && !this.isNumber)
|
|
||||||
throw new Error(`Indexed value must be a string, got type:${typeof(v)}, value:${v}`);
|
|
||||||
}
|
|
||||||
|
|
||||||
prepareValue(v) {
|
|
||||||
let result = v;
|
|
||||||
if (this.numberAsString) {
|
|
||||||
result = v.toString().padStart(this.depth, '0');
|
|
||||||
}
|
|
||||||
if (this.valueAsString && result.length > this.depth)
|
|
||||||
result = result.substring(0, this.depth);
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
add(value, id) {
|
|
||||||
if (value === undefined && this.allowUndef)
|
|
||||||
return;
|
|
||||||
|
|
||||||
this.checkType(value);
|
|
||||||
|
|
||||||
value = this.prepareValue(value);
|
|
||||||
if (this.hash.has(value)) {
|
|
||||||
if (this.unique) {
|
|
||||||
const id_ = this.hash.get(value);
|
|
||||||
if (id_ !== id) {
|
|
||||||
throw new Error(`Collision for unique index detected: value:${value}, id1:${id_}, id2:${id}`);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
const ids = this.hash.get(value);
|
|
||||||
ids.add(id);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if (this.unique) {
|
|
||||||
this.hash.set(value, id);
|
|
||||||
} else {
|
|
||||||
const ids = new Set();
|
|
||||||
this.hash.set(value, ids);
|
|
||||||
ids.add(id);
|
|
||||||
}
|
|
||||||
|
|
||||||
let s = this.sorted.length - 1;
|
|
||||||
const d = this.sorted[s];
|
|
||||||
d.push(value);
|
|
||||||
|
|
||||||
let i = d.length - 1;
|
|
||||||
//вставка
|
|
||||||
while (i > 0 && this.cmp(d[i], d[i - 1]) < 0) {
|
|
||||||
const v = d[i];
|
|
||||||
d[i] = d[i - 1];
|
|
||||||
d[i - 1] = v;
|
|
||||||
i--;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (d.length > 10) {
|
|
||||||
//слияние
|
|
||||||
while (s > 0 && this.sorted[s].length >= this.sorted[s - 1].length) {
|
|
||||||
const a = this.sorted.pop();
|
|
||||||
const b = this.sorted.pop();
|
|
||||||
const c = [];
|
|
||||||
let i = 0;
|
|
||||||
let j = 0;
|
|
||||||
while (i < a.length || j < b.length) {
|
|
||||||
if (i < a.length && (j === b.length || this.cmp(a[i], b[j]) <= 0)) {
|
|
||||||
c.push(a[i]);
|
|
||||||
i++;
|
|
||||||
}
|
|
||||||
if (j < b.length && (i === a.length || this.cmp(b[j], a[i]) <= 0)) {
|
|
||||||
c.push(b[j]);
|
|
||||||
j++;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
this.sorted.push(c);
|
|
||||||
s--;
|
|
||||||
}
|
|
||||||
|
|
||||||
this.sorted.push([]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return value;
|
|
||||||
}
|
|
||||||
|
|
||||||
del(value, id, forceClean = false) {
|
|
||||||
if (value === undefined && this.allowUndef)
|
|
||||||
return;
|
|
||||||
|
|
||||||
this.checkType(value);
|
|
||||||
|
|
||||||
value = this.prepareValue(value);
|
|
||||||
if (this.hash.has(value)) {
|
|
||||||
if (this.unique) {
|
|
||||||
const id_ = this.hash.get(value);
|
|
||||||
if (id_ === id) {
|
|
||||||
this.hash.delete(value);
|
|
||||||
this.delCount++;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
const ids = this.hash.get(value);
|
|
||||||
|
|
||||||
ids.delete(id);
|
|
||||||
|
|
||||||
if (!ids.size) {
|
|
||||||
this.hash.delete(value);
|
|
||||||
this.delCount++;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (this.delCount > (this.sorted[0].length >> 2) || forceClean) {
|
|
||||||
for (let s = 0; s < this.sorted.length; s++) {
|
|
||||||
const a = this.sorted[s];
|
|
||||||
const b = [];
|
|
||||||
for (let i = 0; i < a.length; i++) {
|
|
||||||
if (this.hash.has(a[i]))
|
|
||||||
b.push(a[i]);
|
|
||||||
}
|
|
||||||
this.sorted[s] = b;
|
|
||||||
}
|
|
||||||
|
|
||||||
this.sorted = this.sorted.filter(a => a.length);
|
|
||||||
if (!this.sorted.length) {
|
|
||||||
this.sorted = [[]]
|
|
||||||
} else {
|
|
||||||
this.sorted.sort((a, b) => b.length - a.length);
|
|
||||||
}
|
|
||||||
|
|
||||||
this.delCount = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
return value;
|
|
||||||
}
|
|
||||||
|
|
||||||
reduce(from, to) {
|
|
||||||
const useFrom = (from !== undefined);
|
|
||||||
const useTo = (to !== undefined);
|
|
||||||
|
|
||||||
if (useFrom) {
|
|
||||||
this.checkType(from);
|
|
||||||
from = this.prepareValue(from);
|
|
||||||
}
|
|
||||||
if (useTo) {
|
|
||||||
this.checkType(to);
|
|
||||||
to = this.prepareValue(to);
|
|
||||||
}
|
|
||||||
|
|
||||||
const result = [];
|
|
||||||
for (let s = 0; s < this.sorted.length; s++) {
|
|
||||||
const a = this.sorted[s];
|
|
||||||
if (!a.length) // на всякий случай
|
|
||||||
continue;
|
|
||||||
|
|
||||||
let leftIndex = 0;
|
|
||||||
if (useFrom) {
|
|
||||||
//дихотомия
|
|
||||||
let left = 0;
|
|
||||||
let right = a.length - 1;
|
|
||||||
while (left < right) {
|
|
||||||
let mid = left + ((right - left) >> 1);
|
|
||||||
if (this.cmp(from, a[mid]) <= 0)
|
|
||||||
right = mid;
|
|
||||||
else
|
|
||||||
left = mid + 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
leftIndex = right;
|
|
||||||
if (this.cmp(from, a[right]) > 0)
|
|
||||||
leftIndex++;
|
|
||||||
}
|
|
||||||
|
|
||||||
let rightIndex = a.length;
|
|
||||||
if (useTo) {
|
|
||||||
//дихотомия
|
|
||||||
let left = 0;
|
|
||||||
let right = a.length - 1;
|
|
||||||
while (left < right) {
|
|
||||||
let mid = right - ((right - left) >> 1);
|
|
||||||
if (this.cmp(to, a[mid]) >= 0)
|
|
||||||
left = mid;
|
|
||||||
else
|
|
||||||
right = mid - 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
rightIndex = left;
|
|
||||||
if (this.cmp(to, a[left]) >= 0)
|
|
||||||
rightIndex++;
|
|
||||||
}
|
|
||||||
//console.log(a, leftIndex, rightIndex);
|
|
||||||
if (this.unique) {
|
|
||||||
const ids = new Set();
|
|
||||||
for (let i = leftIndex; i < rightIndex; i++) {
|
|
||||||
const value = a[i];
|
|
||||||
if (this.hash.has(value)) {
|
|
||||||
ids.add(this.hash.get(value));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
result.push(ids);
|
|
||||||
} else {
|
|
||||||
for (let i = leftIndex; i < rightIndex; i++) {
|
|
||||||
const value = a[i];
|
|
||||||
if (this.hash.has(value)) {
|
|
||||||
result.push(this.hash.get(value));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return utils.unionSet(result);
|
|
||||||
}
|
|
||||||
|
|
||||||
min() {
|
|
||||||
let result = new Set();
|
|
||||||
|
|
||||||
let min = null;
|
|
||||||
let id = null;
|
|
||||||
for (let s = 0; s < this.sorted.length; s++) {
|
|
||||||
const a = this.sorted[s];
|
|
||||||
if (!a.length) // на всякий случай
|
|
||||||
continue;
|
|
||||||
if (a[0] < min || min === null) {
|
|
||||||
min = a[0];
|
|
||||||
id = this.hash.get(min);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (id !== null) {
|
|
||||||
if (this.unique)
|
|
||||||
result.add(id);
|
|
||||||
else
|
|
||||||
result = id;
|
|
||||||
}
|
|
||||||
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
max() {
|
|
||||||
let result = new Set();
|
|
||||||
|
|
||||||
let max = null;
|
|
||||||
let id = null;
|
|
||||||
for (let s = 0; s < this.sorted.length; s++) {
|
|
||||||
const a = this.sorted[s];
|
|
||||||
if (!a.length) // на всякий случай
|
|
||||||
continue;
|
|
||||||
|
|
||||||
const last = a.length - 1;
|
|
||||||
if (a[last] > max || max === null) {
|
|
||||||
max = a[last];
|
|
||||||
id = this.hash.get(max);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (id !== null) {
|
|
||||||
if (this.unique)
|
|
||||||
result.add(id);
|
|
||||||
else
|
|
||||||
result = id;
|
|
||||||
}
|
|
||||||
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
iter(checkFunc) {
|
|
||||||
const result = new Set();
|
|
||||||
for (const [value, ids] of this.hash.entries()) {
|
|
||||||
const checkResult = checkFunc(value);
|
|
||||||
if (checkResult === undefined)
|
|
||||||
break;
|
|
||||||
if (checkResult) {
|
|
||||||
if (this.unique) {
|
|
||||||
result.add(ids);
|
|
||||||
} else {
|
|
||||||
for (const id of ids)
|
|
||||||
result.add(id);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
module.exports = TableIndex;
|
|
||||||
File diff suppressed because it is too large
Load Diff
@@ -1,646 +0,0 @@
|
|||||||
const fs = require('fs').promises;
|
|
||||||
const path = require('path');
|
|
||||||
const utils = require('./utils');
|
|
||||||
|
|
||||||
const maxBlockSize = 1024*1024;//bytes
|
|
||||||
|
|
||||||
const minFileDumpSize = 100*1024;//bytes
|
|
||||||
const maxFileDumpSize = 50*1024*1024;//bytes
|
|
||||||
const defragAfter = 10;
|
|
||||||
const defragBlockCountAtOnce = 10;//better >= defragAfter
|
|
||||||
|
|
||||||
class TableRowsFile {
|
|
||||||
constructor(tablePath, cacheSize, compressed) {
|
|
||||||
this.tablePath = tablePath;
|
|
||||||
this.loadedBlocksCount = cacheSize || 5;
|
|
||||||
this.loadedBlocksCount = (this.loadedBlocksCount <= 0 ? 0 : this.loadedBlocksCount);
|
|
||||||
this.compressed = compressed || 0;
|
|
||||||
|
|
||||||
this.blockIndex = new Map();
|
|
||||||
this.currentBlockIndex = 0;
|
|
||||||
this.lastSavedBlockIndex = 0;
|
|
||||||
this.blockList = new Map();
|
|
||||||
this.blocksNotFinalized = new Map();//indexes of blocks
|
|
||||||
this.loadedBlocks = [];
|
|
||||||
this.deltas = new Map();
|
|
||||||
|
|
||||||
this.defragCounter = 0;
|
|
||||||
this.destroyed = false;
|
|
||||||
|
|
||||||
this.blockindex0Size = 0;
|
|
||||||
this.blocklist0Size = 0;
|
|
||||||
|
|
||||||
this.fd = {
|
|
||||||
blockIndex: null,
|
|
||||||
blockList: null,
|
|
||||||
blockRows: null,
|
|
||||||
blockRowsIndex: null,//not a file descriptor
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
//--- rows interface
|
|
||||||
async getRow(id) {
|
|
||||||
const block = this.blockList.get(this.blockIndex.get(id));
|
|
||||||
|
|
||||||
if (block) {
|
|
||||||
if (!block.rows) {
|
|
||||||
await this.loadBlock(block);
|
|
||||||
}
|
|
||||||
|
|
||||||
this.unloadBlocksIfNeeded();//no await
|
|
||||||
return block.rows.get(id);
|
|
||||||
}
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
setRow(id, row, rowStr, deltaStep) {
|
|
||||||
const delta = this.getDelta(deltaStep);
|
|
||||||
|
|
||||||
if (this.blockIndex.has(id)) {
|
|
||||||
this.deleteRow(id, deltaStep, delta);
|
|
||||||
}
|
|
||||||
|
|
||||||
const index = this.addToCurrentBlock(id, row, rowStr, deltaStep, delta);
|
|
||||||
this.blockIndex.set(id, index);
|
|
||||||
delta.blockIndex.push([id, index]);
|
|
||||||
}
|
|
||||||
|
|
||||||
deleteRow(id, deltaStep, delta) {
|
|
||||||
if (this.blockIndex.has(id)) {
|
|
||||||
if (!delta)
|
|
||||||
delta = this.getDelta(deltaStep);
|
|
||||||
|
|
||||||
const block = this.blockList.get(this.blockIndex.get(id));
|
|
||||||
if (block) {
|
|
||||||
block.delCount++;
|
|
||||||
delta.blockList.push([block.index, 1]);
|
|
||||||
}
|
|
||||||
|
|
||||||
this.blockIndex.delete(id);
|
|
||||||
delta.blockIndex.push([id, 0]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
getAllIds() {
|
|
||||||
return this.blockIndex.keys();
|
|
||||||
}
|
|
||||||
|
|
||||||
getAllIdsSize() {
|
|
||||||
return this.blockIndex.size;
|
|
||||||
}
|
|
||||||
//--- rows interface end
|
|
||||||
|
|
||||||
getDelta(deltaStep) {
|
|
||||||
if (this.deltas.has(deltaStep)) {
|
|
||||||
return this.deltas.get(deltaStep);
|
|
||||||
} else {
|
|
||||||
const delta = {
|
|
||||||
blockIndex: [],
|
|
||||||
blockList: [],
|
|
||||||
blockRows: [],
|
|
||||||
};
|
|
||||||
this.deltas.set(deltaStep, delta);
|
|
||||||
return delta;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
createNewBlock() {
|
|
||||||
this.currentBlockIndex++;
|
|
||||||
const block = {
|
|
||||||
index: this.currentBlockIndex,
|
|
||||||
delCount: 0,
|
|
||||||
addCount: 0,
|
|
||||||
size: 0,
|
|
||||||
rows: new Map(),
|
|
||||||
rowsLength: 0,
|
|
||||||
final: false,
|
|
||||||
};
|
|
||||||
this.blockList.set(this.currentBlockIndex, block);
|
|
||||||
this.loadedBlocks.push(this.currentBlockIndex);
|
|
||||||
this.blocksNotFinalized.set(this.currentBlockIndex, 1);
|
|
||||||
|
|
||||||
return block;
|
|
||||||
}
|
|
||||||
|
|
||||||
addToCurrentBlock(id, row, rowStr, deltaStep, delta) {
|
|
||||||
if (!delta)
|
|
||||||
delta = this.getDelta(deltaStep);
|
|
||||||
|
|
||||||
let block = this.blockList.get(this.currentBlockIndex);
|
|
||||||
if (!block)
|
|
||||||
block = this.createNewBlock();
|
|
||||||
|
|
||||||
if (block.size > maxBlockSize)
|
|
||||||
block = this.createNewBlock();
|
|
||||||
|
|
||||||
if (!block.rows) {
|
|
||||||
throw new Error('TableRowsFile: something has gone wrong');
|
|
||||||
}
|
|
||||||
|
|
||||||
block.rows.set(id, row);
|
|
||||||
|
|
||||||
block.addCount++;
|
|
||||||
block.size += rowStr.length;
|
|
||||||
block.rowsLength = block.rows.size;
|
|
||||||
|
|
||||||
delta.blockList.push([block.index, 1]);
|
|
||||||
delta.blockRows.push([block.index, id, row]);
|
|
||||||
|
|
||||||
return block.index;
|
|
||||||
}
|
|
||||||
|
|
||||||
async unloadBlocksIfNeeded() {
|
|
||||||
this.needUnload = true;
|
|
||||||
if (this.unloadingBlocks)
|
|
||||||
return;
|
|
||||||
|
|
||||||
this.unloadingBlocks = true;
|
|
||||||
try {
|
|
||||||
while (this.needUnload) {
|
|
||||||
this.needUnload = false;
|
|
||||||
if (this.destroyed)
|
|
||||||
return;
|
|
||||||
|
|
||||||
await utils.sleep(10);
|
|
||||||
|
|
||||||
//check loaded
|
|
||||||
let missed = new Map();
|
|
||||||
while (this.loadedBlocks.length >= this.loadedBlocksCount) {
|
|
||||||
const index = this.loadedBlocks.shift();
|
|
||||||
if (index >= this.lastSavedBlockIndex) {
|
|
||||||
missed.set(index, 1);
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
const block = this.blockList.get(index);
|
|
||||||
|
|
||||||
if (block) {
|
|
||||||
block.rows = null;
|
|
||||||
//console.log(`unloaded block ${block.index}`);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (this.destroyed)
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
this.loadedBlocks = this.loadedBlocks.concat(Array.from(missed.keys()));
|
|
||||||
}
|
|
||||||
} finally {
|
|
||||||
this.unloadingBlocks = false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async loadFile(filePath) {
|
|
||||||
let buf = await fs.readFile(filePath);
|
|
||||||
if (!buf.length)
|
|
||||||
throw new Error(`TableRowsFile: file ${filePath} is empty`);
|
|
||||||
|
|
||||||
const flag = buf[0];
|
|
||||||
if (flag === 50) {//flag '2' ~ finalized && compressed
|
|
||||||
const packed = Buffer.from(buf.buffer, buf.byteOffset + 1, buf.length - 1);
|
|
||||||
const data = await utils.inflate(packed);
|
|
||||||
buf = data.toString();
|
|
||||||
} else if (flag === 49) {//flag '1' ~ finalized
|
|
||||||
buf[0] = 32;//' '
|
|
||||||
buf = buf.toString();
|
|
||||||
} else {//flag '0' ~ not finalized
|
|
||||||
buf[0] = 32;//' '
|
|
||||||
const last = buf.length - 1;
|
|
||||||
if (buf[last] === 44) {//','
|
|
||||||
buf[last] = 93;//']'
|
|
||||||
buf = buf.toString();
|
|
||||||
} else {//corrupted or empty
|
|
||||||
buf = buf.toString();
|
|
||||||
if (this.loadCorrupted) {
|
|
||||||
const lastComma = buf.lastIndexOf(',');
|
|
||||||
if (lastComma >= 0)
|
|
||||||
buf = buf.substring(0, lastComma);
|
|
||||||
}
|
|
||||||
buf += ']';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
let result;
|
|
||||||
try {
|
|
||||||
result = JSON.parse(buf);
|
|
||||||
} catch(e) {
|
|
||||||
throw new Error(`load ${filePath} failed: ${e.message}`);
|
|
||||||
}
|
|
||||||
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
async writeFinal(fileName, data) {
|
|
||||||
if (!this.compressed) {
|
|
||||||
await fs.writeFile(fileName, '1' + data);
|
|
||||||
} else {
|
|
||||||
let buf = Buffer.from(data);
|
|
||||||
buf = await utils.deflate(buf, this.compressed);
|
|
||||||
const fd = await fs.open(fileName, 'w');
|
|
||||||
await fd.write('2');
|
|
||||||
await fd.write(buf);
|
|
||||||
await fd.close();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async loadBlock(block) {
|
|
||||||
//console.log(`start load block ${block.index}`);
|
|
||||||
if (!block.rows) {
|
|
||||||
const arr = await this.loadFile(this.blockRowsFilePath(block.index));
|
|
||||||
|
|
||||||
block.rows = new Map(arr);
|
|
||||||
|
|
||||||
this.loadedBlocks.push(block.index);
|
|
||||||
//console.log(`loaded block ${block.index}`);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async closeFd(name) {
|
|
||||||
if (this.fd[name]) {
|
|
||||||
await this.fd[name].close();
|
|
||||||
this.fd[name] = null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async openFd(name, fileName = '') {
|
|
||||||
if (this.fd[name])
|
|
||||||
return;
|
|
||||||
|
|
||||||
if (!fileName) {
|
|
||||||
throw new Error('TableRowsFile: fileName is empty');
|
|
||||||
}
|
|
||||||
|
|
||||||
const exists = await utils.pathExists(fileName);
|
|
||||||
|
|
||||||
const fd = await fs.open(fileName, 'a');
|
|
||||||
if (!exists) {
|
|
||||||
await fd.write('0[');
|
|
||||||
}
|
|
||||||
|
|
||||||
this.fd[name] = fd;
|
|
||||||
}
|
|
||||||
|
|
||||||
blockRowsFilePath(index) {
|
|
||||||
if (index < 1000000)
|
|
||||||
return `${this.tablePath}/${index.toString().padStart(6, '0')}.jem`;
|
|
||||||
else
|
|
||||||
return `${this.tablePath}/${index.toString().padStart(12, '0')}.jem`;
|
|
||||||
}
|
|
||||||
|
|
||||||
async finalizeBlocks() {
|
|
||||||
//console.log(this.blocksNotFinalized.size);
|
|
||||||
|
|
||||||
for (const index of this.blocksNotFinalized.keys()) {
|
|
||||||
if (this.destroyed)
|
|
||||||
return;
|
|
||||||
|
|
||||||
if (index >= this.lastSavedBlockIndex)
|
|
||||||
continue;
|
|
||||||
|
|
||||||
const block = this.blockList.get(index);
|
|
||||||
|
|
||||||
if (block) {
|
|
||||||
if (block.final)
|
|
||||||
throw new Error('finalizeBlocks: something wrong');
|
|
||||||
|
|
||||||
const blockPath = this.blockRowsFilePath(block.index);
|
|
||||||
//console.log(`start finalize block ${block.index}`);
|
|
||||||
const arr = await this.loadFile(blockPath);
|
|
||||||
const rows = new Map(arr);
|
|
||||||
|
|
||||||
const finBlockPath = `${blockPath}.tmp`;
|
|
||||||
const rowsStr = JSON.stringify(Array.from(rows));
|
|
||||||
await this.writeFinal(finBlockPath, rowsStr);
|
|
||||||
|
|
||||||
await fs.rename(finBlockPath, blockPath);
|
|
||||||
|
|
||||||
block.size = Buffer.byteLength(rowsStr, 'utf8') + 1;
|
|
||||||
block.rowsLength = rows.size;//insurance
|
|
||||||
block.final = true;
|
|
||||||
await this.fd.blockList.write(JSON.stringify(block) + ',');
|
|
||||||
//console.log(`finalized block ${block.index}`);
|
|
||||||
}
|
|
||||||
|
|
||||||
this.blocksNotFinalized.delete(index);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async dumpMaps() {
|
|
||||||
//dumping blockIndex
|
|
||||||
const blockindex1Size = (await this.fd.blockIndex.stat()).size;
|
|
||||||
if ((blockindex1Size > minFileDumpSize && blockindex1Size > this.blockindex0Size) || blockindex1Size > maxFileDumpSize) {
|
|
||||||
const blockindex0Path = `${this.tablePath}/blockindex.0`;
|
|
||||||
const blockindex2Path = `${this.tablePath}/blockindex.2`;
|
|
||||||
await this.writeFinal(blockindex2Path, JSON.stringify(Array.from(this.blockIndex)));
|
|
||||||
|
|
||||||
await fs.rename(blockindex2Path, blockindex0Path);
|
|
||||||
await this.closeFd('blockIndex');
|
|
||||||
await fs.unlink(`${this.tablePath}/blockindex.1`);
|
|
||||||
this.blockindex0Size = (await fs.stat(blockindex0Path)).size;
|
|
||||||
}
|
|
||||||
|
|
||||||
//dumping blockList
|
|
||||||
const blocklist1Size = (await this.fd.blockList.stat()).size;
|
|
||||||
if ((blocklist1Size > minFileDumpSize && blocklist1Size > this.blocklist0Size) || blocklist1Size > maxFileDumpSize) {
|
|
||||||
const blocklist0Path = `${this.tablePath}/blocklist.0`;
|
|
||||||
const blocklist2Path = `${this.tablePath}/blocklist.2`;
|
|
||||||
await this.writeFinal(blocklist2Path, JSON.stringify(Array.from(this.blockList.values())));
|
|
||||||
|
|
||||||
await fs.rename(blocklist2Path, blocklist0Path);
|
|
||||||
await this.closeFd('blockList');
|
|
||||||
await fs.unlink(`${this.tablePath}/blocklist.1`);
|
|
||||||
this.blocklist0Size = (await fs.stat(blocklist0Path)).size;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async saveDelta(deltaStep) {
|
|
||||||
const delta = this.getDelta(deltaStep);
|
|
||||||
|
|
||||||
//lastSavedBlockIndex
|
|
||||||
const len = delta.blockRows.length;
|
|
||||||
if (len) {
|
|
||||||
this.lastSavedBlockIndex = delta.blockRows[len - 1][0];
|
|
||||||
}
|
|
||||||
|
|
||||||
//check all blocks fragmentation
|
|
||||||
if (!this.defragCandidates)
|
|
||||||
this.defragCandidates = [];
|
|
||||||
|
|
||||||
if (!this.defragCandidates.length) {
|
|
||||||
if (this.defragCounter >= defragAfter) {
|
|
||||||
for (const block of this.blockList.values()) {
|
|
||||||
if (!block.final)
|
|
||||||
continue;
|
|
||||||
|
|
||||||
if (block.addCount - block.delCount < block.rowsLength/2 || block.size < maxBlockSize/2) {
|
|
||||||
this.defragCandidates.push(block);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
this.defragCounter = 0;
|
|
||||||
} else {
|
|
||||||
this.defragCounter++;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
let defragmented = 0;
|
|
||||||
while (this.defragCandidates.length) {
|
|
||||||
if (defragmented >= defragBlockCountAtOnce || this.destroyed)
|
|
||||||
break;
|
|
||||||
|
|
||||||
const block = this.defragCandidates.shift();
|
|
||||||
|
|
||||||
if (!block.rows) {
|
|
||||||
await this.loadBlock(block);
|
|
||||||
}
|
|
||||||
|
|
||||||
//move all active rows from fragmented block to current
|
|
||||||
for (const [id, row] of block.rows.entries()) {
|
|
||||||
if (this.blockIndex.get(id) === block.index) {
|
|
||||||
const newIndex = this.addToCurrentBlock(id, row, JSON.stringify(row), deltaStep, delta);
|
|
||||||
this.blockIndex.set(id, newIndex);
|
|
||||||
delta.blockIndex.push([id, newIndex]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
this.blockList.delete(block.index);
|
|
||||||
delta.blockList.push([block.index, 0]);
|
|
||||||
|
|
||||||
if (!delta.delFiles)
|
|
||||||
delta.delFiles = [];
|
|
||||||
delta.delFiles.push(this.blockRowsFilePath(block.index));
|
|
||||||
|
|
||||||
defragmented++;
|
|
||||||
//console.log(`defragmented block ${block.index}, size: ${block.size}, addCount: ${block.addCount}, delCount: ${block.delCount}, rowsLength: ${block.rowsLength}`);
|
|
||||||
}
|
|
||||||
|
|
||||||
//blockIndex delta save
|
|
||||||
if (!this.fd.blockIndex)
|
|
||||||
await this.openFd('blockIndex', `${this.tablePath}/blockindex.1`);
|
|
||||||
|
|
||||||
let buf = [];
|
|
||||||
for (const deltaRec of delta.blockIndex) {
|
|
||||||
buf.push(JSON.stringify(deltaRec));
|
|
||||||
}
|
|
||||||
if (buf.length)
|
|
||||||
await this.fd.blockIndex.write(buf.join(',') + ',');
|
|
||||||
|
|
||||||
//blockList delta save
|
|
||||||
if (!this.fd.blockList)
|
|
||||||
await this.openFd('blockList', `${this.tablePath}/blocklist.1`);
|
|
||||||
|
|
||||||
let lastSaved = 0;
|
|
||||||
buf = [];
|
|
||||||
for (const deltaRec of delta.blockList) {
|
|
||||||
const index = deltaRec[0];
|
|
||||||
const exists = deltaRec[1];
|
|
||||||
|
|
||||||
if (exists) {
|
|
||||||
if (lastSaved !== index) {//optimization
|
|
||||||
const block = this.blockList.get(index);
|
|
||||||
if (block)//might be defragmented already
|
|
||||||
buf.push(JSON.stringify(block));
|
|
||||||
lastSaved = index;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
buf.push(JSON.stringify({index, deleted: 1}));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (buf.length)
|
|
||||||
await this.fd.blockList.write(buf.join(',') + ',');
|
|
||||||
|
|
||||||
//blockRows delta save
|
|
||||||
buf = [];
|
|
||||||
for (const deltaRec of delta.blockRows) {
|
|
||||||
const [index, id, row] = deltaRec;
|
|
||||||
|
|
||||||
if (this.fd.blockRowsIndex !== index) {
|
|
||||||
if (buf.length)
|
|
||||||
await this.fd.blockRows.write(buf.join(',') + ',');
|
|
||||||
buf = [];
|
|
||||||
await this.closeFd('blockRows');
|
|
||||||
this.fd.blockRowsIndex = null;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!this.fd.blockRows) {
|
|
||||||
const blockPath = this.blockRowsFilePath(index);
|
|
||||||
|
|
||||||
await this.openFd('blockRows', blockPath);
|
|
||||||
this.fd.blockRowsIndex = index;
|
|
||||||
}
|
|
||||||
|
|
||||||
buf.push(JSON.stringify([id, row]));
|
|
||||||
}
|
|
||||||
if (buf.length)
|
|
||||||
await this.fd.blockRows.write(buf.join(',') + ',');
|
|
||||||
|
|
||||||
//blocks finalization
|
|
||||||
await this.finalizeBlocks();
|
|
||||||
this.unloadBlocksIfNeeded();//no await
|
|
||||||
|
|
||||||
//dumps if needed
|
|
||||||
await this.dumpMaps();
|
|
||||||
|
|
||||||
//delete files if needed
|
|
||||||
if (delta.delFiles) {
|
|
||||||
for (const fileName of delta.delFiles) {
|
|
||||||
//console.log(`delete ${fileName}`);
|
|
||||||
if (await utils.pathExists(fileName))
|
|
||||||
await fs.unlink(fileName);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
this.deltas.delete(deltaStep);
|
|
||||||
}
|
|
||||||
|
|
||||||
async cancelDelta(deltaStep) {
|
|
||||||
this.deltas.delete(deltaStep);
|
|
||||||
}
|
|
||||||
|
|
||||||
async load() {
|
|
||||||
let autoIncrement = 0;
|
|
||||||
|
|
||||||
const loadBlockIndex = (fileNum, data) => {
|
|
||||||
if (fileNum === 0) {//dumped data
|
|
||||||
this.blockIndex = new Map(data);//much faster
|
|
||||||
for (const id of this.blockIndex.keys()) {
|
|
||||||
if (typeof(id) === 'number' && id >= autoIncrement)
|
|
||||||
autoIncrement = id + 1;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
for (const rec of data) {
|
|
||||||
const [id, index] = rec;
|
|
||||||
if (index > 0) {
|
|
||||||
this.blockIndex.set(id, index);
|
|
||||||
if (typeof(id) === 'number' && id >= autoIncrement)
|
|
||||||
autoIncrement = id + 1;
|
|
||||||
} else
|
|
||||||
this.blockIndex.delete(id);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const loadBlockList = (data) => {
|
|
||||||
for (const rec of data) {
|
|
||||||
const block = rec;
|
|
||||||
if (block.deleted) {
|
|
||||||
this.blockList.delete(block.index);
|
|
||||||
} else {
|
|
||||||
block.rows = null;
|
|
||||||
this.blockList.set(block.index, block);
|
|
||||||
if (block.index > this.currentBlockIndex)
|
|
||||||
this.currentBlockIndex = block.index;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
this.blockIndex.clear();
|
|
||||||
for (let i = 0; i < 2; i++) {
|
|
||||||
const dataPath = `${this.tablePath}/blockindex.${i}`;
|
|
||||||
|
|
||||||
if (await utils.pathExists(dataPath)) {
|
|
||||||
const data = await this.loadFile(dataPath);
|
|
||||||
loadBlockIndex(i, data);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
const blockindex0Path = `${this.tablePath}/blockindex.0`;
|
|
||||||
if (await utils.pathExists(blockindex0Path))
|
|
||||||
this.blockindex0Size = (await fs.stat(blockindex0Path)).size;
|
|
||||||
|
|
||||||
this.currentBlockIndex = 0;
|
|
||||||
this.blockList.clear();
|
|
||||||
for (let i = 0; i < 2; i++) {
|
|
||||||
const dataPath = `${this.tablePath}/blocklist.${i}`;
|
|
||||||
|
|
||||||
if (await utils.pathExists(dataPath)) {
|
|
||||||
const data = await this.loadFile(dataPath);
|
|
||||||
loadBlockList(data);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
const blocklist0Path = `${this.tablePath}/blocklist.0`;
|
|
||||||
if (await utils.pathExists(blocklist0Path))
|
|
||||||
this.blocklist0Size = (await fs.stat(blocklist0Path)).size;
|
|
||||||
|
|
||||||
this.lastSavedBlockIndex = this.currentBlockIndex;
|
|
||||||
const currentBlock = this.blockList.get(this.currentBlockIndex);
|
|
||||||
if (currentBlock)
|
|
||||||
await this.loadBlock(currentBlock);
|
|
||||||
|
|
||||||
this.blocksNotFinalized = new Map();
|
|
||||||
for (const block of this.blockList.values()) {
|
|
||||||
if (!block.final)
|
|
||||||
this.blocksNotFinalized.set(block.index, 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
return autoIncrement;
|
|
||||||
}
|
|
||||||
|
|
||||||
async loadCorrupted() {
|
|
||||||
this.loadCorrupted = true;
|
|
||||||
|
|
||||||
const loadBlockIndex = (fileNum, data) => {
|
|
||||||
if (fileNum === 0) {//dumped data
|
|
||||||
this.blockIndex = new Map(data);//much faster
|
|
||||||
} else {
|
|
||||||
for (const rec of data) {
|
|
||||||
const [id, index] = rec;
|
|
||||||
if (index > 0)
|
|
||||||
this.blockIndex.set(id, index);
|
|
||||||
else
|
|
||||||
this.blockIndex.delete(id);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
this.blockIndex.clear();
|
|
||||||
for (let i = 0; i < 2; i++) {
|
|
||||||
const dataPath = `${this.tablePath}/blockindex.${i}`;
|
|
||||||
|
|
||||||
if (await utils.pathExists(dataPath)) {
|
|
||||||
try {
|
|
||||||
const data = await this.loadFile(dataPath);
|
|
||||||
loadBlockIndex(i, data);
|
|
||||||
} catch(e) {
|
|
||||||
console.error(e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const files = await fs.readdir(this.tablePath, { withFileTypes: true });
|
|
||||||
|
|
||||||
this.blockList.clear();
|
|
||||||
for (const file of files) {
|
|
||||||
if (file.isFile() && path.extname(file.name) == '.jem') {
|
|
||||||
const numStr = path.basename(file.name, '.jem');
|
|
||||||
const index = parseInt(numStr, 10);
|
|
||||||
if (!isNaN(index)) {
|
|
||||||
const block = {
|
|
||||||
index,
|
|
||||||
delCount: 0,
|
|
||||||
addCount: 0,
|
|
||||||
size: 0,
|
|
||||||
rows: null,
|
|
||||||
rowsLength: 0,
|
|
||||||
final: false,
|
|
||||||
};
|
|
||||||
this.blockList.set(block.index, block);
|
|
||||||
//console.log(index);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async closeAllFiles() {
|
|
||||||
await this.closeFd('blockIndex');
|
|
||||||
await this.closeFd('blockList');
|
|
||||||
await this.closeFd('blockRows');
|
|
||||||
}
|
|
||||||
|
|
||||||
async destroy() {
|
|
||||||
await this.closeAllFiles();
|
|
||||||
|
|
||||||
this.destroyed = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
module.exports = TableRowsFile;
|
|
||||||
@@ -1,34 +0,0 @@
|
|||||||
class TableRowsMem {
|
|
||||||
constructor() {
|
|
||||||
this.rows = new Map();
|
|
||||||
}
|
|
||||||
|
|
||||||
//--- rows interface
|
|
||||||
async getRow(id) {
|
|
||||||
return this.rows.get(id);
|
|
||||||
}
|
|
||||||
|
|
||||||
setRow(id, row) {
|
|
||||||
this.rows.set(id, row);
|
|
||||||
}
|
|
||||||
|
|
||||||
deleteRow(id) {
|
|
||||||
this.rows.delete(id);
|
|
||||||
}
|
|
||||||
|
|
||||||
getAllIds() {
|
|
||||||
return this.rows.keys();
|
|
||||||
}
|
|
||||||
|
|
||||||
getAllIdsSize() {
|
|
||||||
return this.rows.size;
|
|
||||||
}
|
|
||||||
//--- rows interface end
|
|
||||||
|
|
||||||
async destroy() {
|
|
||||||
//for GC
|
|
||||||
this.rows = null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
module.exports = TableRowsMem;
|
|
||||||
@@ -1,7 +0,0 @@
|
|||||||
const JembaDb = require('./JembaDb');
|
|
||||||
const JembaDbThread = require('./JembaDbThread');
|
|
||||||
|
|
||||||
module.exports = {
|
|
||||||
JembaDb,
|
|
||||||
JembaDbThread,
|
|
||||||
};
|
|
||||||
@@ -1,152 +0,0 @@
|
|||||||
const fsCB = require('fs');
|
|
||||||
const fs = fsCB.promises;
|
|
||||||
const zlib = require('zlib');
|
|
||||||
|
|
||||||
function sleep(ms) {
|
|
||||||
return new Promise(resolve => setTimeout(resolve, ms));
|
|
||||||
}
|
|
||||||
|
|
||||||
function sleepWithStop(ms, cb = () => {}) {
|
|
||||||
return new Promise(resolve => {
|
|
||||||
const timer = setTimeout(resolve, ms);
|
|
||||||
cb(() => { clearTimeout(timer); resolve(); });
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
function unionSet(arrSet) {
|
|
||||||
if (!arrSet.length)
|
|
||||||
return new Set();
|
|
||||||
|
|
||||||
let max = 0;
|
|
||||||
let size = arrSet[0].size;
|
|
||||||
for (let i = 1; i < arrSet.length; i++) {
|
|
||||||
if (arrSet[i].size > size) {
|
|
||||||
max = i;
|
|
||||||
size = arrSet[i].size;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const result = new Set(arrSet[max]);
|
|
||||||
for (let i = 0; i < arrSet.length; i++) {
|
|
||||||
if (i === max)
|
|
||||||
continue;
|
|
||||||
|
|
||||||
for (const elem of arrSet[i]) {
|
|
||||||
result.add(elem);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
function intersectSet(arrSet) {
|
|
||||||
if (!arrSet.length)
|
|
||||||
return new Set();
|
|
||||||
|
|
||||||
let min = 0;
|
|
||||||
let size = arrSet[0].size;
|
|
||||||
for (let i = 1; i < arrSet.length; i++) {
|
|
||||||
if (arrSet[i].size < size) {
|
|
||||||
min = i;
|
|
||||||
size = arrSet[i].size;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const result = new Set();
|
|
||||||
for (const elem of arrSet[min]) {
|
|
||||||
let inAll = true;
|
|
||||||
for (let i = 0; i < arrSet.length; i++) {
|
|
||||||
if (i === min)
|
|
||||||
continue;
|
|
||||||
if (!arrSet[i].has(elem)) {
|
|
||||||
inAll = false;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (inAll)
|
|
||||||
result.add(elem);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
async function pathExists(path) {
|
|
||||||
try {
|
|
||||||
await fs.access(path);
|
|
||||||
return true;
|
|
||||||
} catch(e) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async function appendFileToFile(nameFrom, nameTo) {
|
|
||||||
return new Promise((resolve, reject) => {
|
|
||||||
const readStream = fsCB.createReadStream(nameFrom);
|
|
||||||
readStream.on('error', (err) => {
|
|
||||||
reject(err);
|
|
||||||
});
|
|
||||||
|
|
||||||
const writeStream = fsCB.createWriteStream(nameTo, {flags: 'a'});
|
|
||||||
|
|
||||||
writeStream.on('error', (err) => {
|
|
||||||
reject(err);
|
|
||||||
});
|
|
||||||
|
|
||||||
writeStream.on('close', () => {
|
|
||||||
resolve();
|
|
||||||
});
|
|
||||||
|
|
||||||
readStream.pipe(writeStream);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
function esc(obj) {
|
|
||||||
return JSON.stringify(obj).replace(/@/g, '\\x40');
|
|
||||||
}
|
|
||||||
|
|
||||||
function paramToArray(param) {
|
|
||||||
return (Array.isArray(param) ? param : [param]);
|
|
||||||
}
|
|
||||||
|
|
||||||
function cloneDeep(obj) {
|
|
||||||
return JSON.parse(JSON.stringify(obj));
|
|
||||||
}
|
|
||||||
|
|
||||||
//async
|
|
||||||
function deflate(buf, compressionLevel) {
|
|
||||||
return new Promise((resolve, reject) => {
|
|
||||||
zlib.deflateRaw(buf, {level: compressionLevel}, (err, b) => {
|
|
||||||
if (err)
|
|
||||||
reject(err);
|
|
||||||
resolve(b);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
//async
|
|
||||||
function inflate(buf) {
|
|
||||||
return new Promise((resolve, reject) => {
|
|
||||||
zlib.inflateRaw(buf, (err, b) => {
|
|
||||||
if (err)
|
|
||||||
reject(err);
|
|
||||||
resolve(b);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
module.exports = {
|
|
||||||
sleep,
|
|
||||||
sleepWithStop,
|
|
||||||
unionSet,
|
|
||||||
intersectSet,
|
|
||||||
pathExists,
|
|
||||||
appendFileToFile,
|
|
||||||
esc,
|
|
||||||
paramToArray,
|
|
||||||
cloneDeep,
|
|
||||||
deflate,
|
|
||||||
inflate,
|
|
||||||
};
|
|
||||||
@@ -8,10 +8,11 @@ const http = require('http');
|
|||||||
const WebSocket = require ('ws');
|
const WebSocket = require ('ws');
|
||||||
|
|
||||||
const ayncExit = new (require('./core/AsyncExit'))();
|
const ayncExit = new (require('./core/AsyncExit'))();
|
||||||
ayncExit.init();
|
|
||||||
|
|
||||||
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
|
||||||
@@ -64,7 +65,7 @@ async function main() {
|
|||||||
if (serverCfg.mode !== 'none') {
|
if (serverCfg.mode !== 'none') {
|
||||||
const app = express();
|
const app = express();
|
||||||
const server = http.createServer(app);
|
const server = http.createServer(app);
|
||||||
const wss = new WebSocket.Server({ server, maxPayload: 10*1024*1024 });
|
const wss = new WebSocket.Server({ server, maxPayload: maxPayloadSize*1024*1024 });
|
||||||
|
|
||||||
const serverConfig = Object.assign({}, config, serverCfg);
|
const serverConfig = Object.assign({}, config, serverCfg);
|
||||||
|
|
||||||
@@ -76,7 +77,7 @@ 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);
|
||||||
|
|
||||||
|
|||||||
@@ -31,7 +31,6 @@ function initRoutes(app, wss, config) {
|
|||||||
['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/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], {}],
|
||||||
['POST', '/api/worker/get-state-finish', worker.getStateFinish.bind(worker), [aAll], {}],
|
|
||||||
];
|
];
|
||||||
|
|
||||||
//to app
|
//to app
|
||||||
|
|||||||
Reference in New Issue
Block a user