reader.js 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124
  1. import _ from 'lodash';
  2. import axios from 'axios';
  3. import {Buffer} from 'safe-buffer';
  4. import * as utils from '../share/utils';
  5. const api = axios.create({
  6. baseURL: '/api/reader'
  7. });
  8. const workerApi = axios.create({
  9. baseURL: '/api/worker'
  10. });
  11. class Reader {
  12. async loadBook(url, callback) {
  13. const refreshPause = 300;
  14. if (!callback) callback = () => {};
  15. let response = await api.post('/load-book', {type: 'url', url});
  16. const workerId = response.data.workerId;
  17. if (!workerId)
  18. throw new Error('Неверный ответ api');
  19. callback({totalSteps: 4});
  20. let i = 0;
  21. while (1) {// eslint-disable-line no-constant-condition
  22. callback(response.data);
  23. if (response.data.state == 'finish') {//воркер закончил работу, можно скачивать кешированный на сервере файл
  24. callback({step: 4});
  25. const book = await this.loadCachedBook(response.data.path, callback);
  26. return Object.assign({}, response.data, {data: book.data});
  27. }
  28. if (response.data.state == 'error') {
  29. let errMes = response.data.error;
  30. if (errMes.indexOf('getaddrinfo') >= 0 ||
  31. errMes.indexOf('ECONNRESET') >= 0 ||
  32. errMes.indexOf('EINVAL') >= 0 ||
  33. errMes.indexOf('404') >= 0)
  34. errMes = `Ресурс не найден по адресу: ${response.data.url}`;
  35. throw new Error(errMes);
  36. }
  37. if (i > 0)
  38. await utils.sleep(refreshPause);
  39. i++;
  40. if (i > 120*1000/refreshPause) {//2 мин ждем телодвижений воркера
  41. throw new Error('Слишком долгое время ожидания');
  42. }
  43. //проверка воркера
  44. const prevProgress = response.data.progress;
  45. const prevState = response.data.state;
  46. response = await workerApi.post('/get-state', {workerId});
  47. i = (prevProgress != response.data.progress || prevState != response.data.state ? 1 : i);
  48. }
  49. }
  50. async loadCachedBook(url, callback){
  51. const response = await axios.head(url);
  52. let estSize = 1000000;
  53. if (response.headers['content-length']) {
  54. estSize = response.headers['content-length'];
  55. }
  56. const options = {
  57. onDownloadProgress: progress => {
  58. while (progress.loaded > estSize) estSize *= 1.5;
  59. if (callback)
  60. callback({state: 'loading', progress: Math.round((progress.loaded*100)/estSize)});
  61. }
  62. }
  63. //загрузка
  64. return await axios.get(url, options);
  65. }
  66. async uploadFile(file, maxUploadFileSize, callback) {
  67. if (!maxUploadFileSize)
  68. maxUploadFileSize = 10*1024*1024;
  69. if (file.size > maxUploadFileSize)
  70. throw new Error(`Размер файла превышает ${maxUploadFileSize} байт`);
  71. let formData = new FormData();
  72. formData.append('file', file);
  73. const options = {
  74. headers: {
  75. 'Content-Type': 'multipart/form-data'
  76. },
  77. onUploadProgress: progress => {
  78. const total = (progress.total ? progress.total : progress.loaded + 200000);
  79. if (callback)
  80. callback({state: 'upload', progress: Math.round((progress.loaded*100)/total)});
  81. }
  82. };
  83. let response = await api.post('/upload-file', formData, options);
  84. if (response.data.state == 'error')
  85. throw new Error(response.data.error);
  86. const url = response.data.url;
  87. if (!url)
  88. throw new Error('Неверный ответ api');
  89. return url;
  90. }
  91. async storage(request) {
  92. let response = await api.post('/storage', request);
  93. const state = response.data.state;
  94. if (!state)
  95. throw new Error('Неверный ответ api');
  96. return response.data;
  97. }
  98. }
  99. export default new Reader();