123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124 |
- import _ from 'lodash';
- import axios from 'axios';
- import {Buffer} from 'safe-buffer';
- import * as utils from '../share/utils';
- const api = axios.create({
- baseURL: '/api/reader'
- });
- const workerApi = axios.create({
- baseURL: '/api/worker'
- });
- class Reader {
- async loadBook(url, callback) {
- const refreshPause = 300;
- if (!callback) callback = () => {};
- let response = await api.post('/load-book', {type: 'url', url});
- const workerId = response.data.workerId;
- if (!workerId)
- throw new Error('Неверный ответ api');
- callback({totalSteps: 4});
- let i = 0;
- while (1) {// eslint-disable-line no-constant-condition
- callback(response.data);
- if (response.data.state == 'finish') {//воркер закончил работу, можно скачивать кешированный на сервере файл
- callback({step: 4});
- const book = await this.loadCachedBook(response.data.path, callback);
- return Object.assign({}, response.data, {data: book.data});
- }
- if (response.data.state == 'error') {
- let errMes = response.data.error;
- if (errMes.indexOf('getaddrinfo') >= 0 ||
- errMes.indexOf('ECONNRESET') >= 0 ||
- errMes.indexOf('EINVAL') >= 0 ||
- errMes.indexOf('404') >= 0)
- errMes = `Ресурс не найден по адресу: ${response.data.url}`;
- throw new Error(errMes);
- }
- if (i > 0)
- await utils.sleep(refreshPause);
- i++;
- if (i > 120*1000/refreshPause) {//2 мин ждем телодвижений воркера
- throw new Error('Слишком долгое время ожидания');
- }
- //проверка воркера
- const prevProgress = response.data.progress;
- const prevState = response.data.state;
- response = await workerApi.post('/get-state', {workerId});
- i = (prevProgress != response.data.progress || prevState != response.data.state ? 1 : i);
- }
- }
- async loadCachedBook(url, callback){
- const response = await axios.head(url);
- let estSize = 1000000;
- if (response.headers['content-length']) {
- estSize = response.headers['content-length'];
- }
- const options = {
- onDownloadProgress: progress => {
- while (progress.loaded > estSize) estSize *= 1.5;
- if (callback)
- callback({state: 'loading', progress: Math.round((progress.loaded*100)/estSize)});
- }
- }
- //загрузка
- return await axios.get(url, options);
- }
- async uploadFile(file, maxUploadFileSize, callback) {
- if (!maxUploadFileSize)
- maxUploadFileSize = 10*1024*1024;
- if (file.size > maxUploadFileSize)
- throw new Error(`Размер файла превышает ${maxUploadFileSize} байт`);
- let formData = new FormData();
- formData.append('file', file);
- const options = {
- headers: {
- 'Content-Type': 'multipart/form-data'
- },
- onUploadProgress: progress => {
- const total = (progress.total ? progress.total : progress.loaded + 200000);
- if (callback)
- callback({state: 'upload', progress: Math.round((progress.loaded*100)/total)});
- }
- };
- let response = await api.post('/upload-file', formData, options);
- if (response.data.state == 'error')
- throw new Error(response.data.error);
- const url = response.data.url;
- if (!url)
- throw new Error('Неверный ответ api');
- return url;
- }
- async storage(request) {
- let response = await api.post('/storage', request);
- const state = response.data.state;
- if (!state)
- throw new Error('Неверный ответ api');
- return response.data;
- }
- }
- export default new Reader();
|