ConvertDjvu.js 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111
  1. const fs = require('fs-extra');
  2. const path = require('path');
  3. const utils = require('../../utils');
  4. const ConvertJpegPng = require('./ConvertJpegPng');
  5. class ConvertDjvu extends ConvertJpegPng {
  6. check(data, opts) {
  7. const {inputFiles} = opts;
  8. return this.config.useExternalBookConverter &&
  9. inputFiles.sourceFileType && inputFiles.sourceFileType.ext == 'djvu';
  10. }
  11. async run(data, opts) {
  12. if (!this.check(data, opts))
  13. return false;
  14. let {inputFiles, callback, abort, djvuQuality} = opts;
  15. djvuQuality = (djvuQuality && djvuQuality <= 100 && djvuQuality >= 10 ? djvuQuality : 20);
  16. let jpegQuality = djvuQuality;
  17. let tiffQuality = djvuQuality + 30;
  18. tiffQuality = (tiffQuality < 85 ? tiffQuality : 85);
  19. const ddjvuPath = '/usr/bin/ddjvu';
  20. if (!await fs.pathExists(ddjvuPath))
  21. throw new Error('Внешний конвертер ddjvu не найден');
  22. const djvusedPath = '/usr/bin/djvused';
  23. if (!await fs.pathExists(djvusedPath))
  24. throw new Error('Внешний конвертер djvused не найден');
  25. const tiffsplitPath = '/usr/bin/tiffsplit';
  26. if (!await fs.pathExists(tiffsplitPath))
  27. throw new Error('Внешний конвертер tiffsplitPath не найден');
  28. const mogrifyPath = '/usr/bin/mogrify';
  29. if (!await fs.pathExists(mogrifyPath))
  30. throw new Error('Внешний конвертер mogrifyPath не найден');
  31. const dir = `${inputFiles.filesDir}/`;
  32. const baseFile = `${dir}${path.basename(inputFiles.sourceFile)}`;
  33. const tifFile = `${baseFile}.tif`;
  34. //конвертируем в tiff
  35. let perc = 0;
  36. await this.execConverter(ddjvuPath, ['-format=tiff', `-quality=${tiffQuality}`, '-verbose', inputFiles.sourceFile, tifFile], () => {
  37. perc = (perc < 100 ? perc + 1 : 40);
  38. callback(perc);
  39. }, abort);
  40. const tifFileSize = (await fs.stat(tifFile)).size;
  41. let limitSize = 4*this.config.maxUploadFileSize;
  42. if (tifFileSize > limitSize) {
  43. throw new Error(`Файл для конвертирования слишком большой|FORLOG| tifFileSize: ${tifFileSize} > ${limitSize}`);
  44. }
  45. //разбиваем на файлы
  46. await this.execConverter(tiffsplitPath, [tifFile, dir], null, abort);
  47. await fs.remove(tifFile);
  48. //конвертируем в jpg
  49. await this.execConverter(mogrifyPath, ['-quality', jpegQuality, '-scale', '2048>', '-verbose', '-format', 'jpg', `${dir}*.tif`], () => {
  50. perc = (perc < 100 ? perc + 1 : 40);
  51. callback(perc);
  52. }, abort);
  53. limitSize = 2*this.config.maxUploadFileSize;
  54. let jpgFilesSize = 0;
  55. //ищем изображения
  56. let files = [];
  57. await utils.findFiles(async(file) => {
  58. if (path.extname(file) == '.jpg') {
  59. jpgFilesSize += (await fs.stat(file)).size;
  60. if (jpgFilesSize > limitSize) {
  61. throw new Error(`Файл для конвертирования слишком большой|FORLOG| jpgFilesSize: ${jpgFilesSize} > ${limitSize}`);
  62. }
  63. files.push({name: file, base: path.basename(file)});
  64. }
  65. }, dir);
  66. files.sort((a, b) => a.base.localeCompare(b.base));
  67. //схема документа (outline)
  68. const djvusedResult = await this.execConverter(djvusedPath, ['-u', '-e', 'print-outline', inputFiles.sourceFile]);
  69. const outline = [];
  70. const lines = djvusedResult.stdout.match(/\(".*"\s*?"#\d+".*?\)/g);
  71. if (lines) {
  72. lines.forEach(l => {
  73. const m = l.match(/"(.*)"\s*?"#(\d+)"/);
  74. if (m) {
  75. outline[m[2]] = m[1];
  76. }
  77. });
  78. }
  79. await utils.sleep(100);
  80. let i = 0;
  81. const imageFiles = files.map(f => {
  82. i++;
  83. let alt = (outline[i] ? outline[i] : '');
  84. return {src: f.name, alt};
  85. });
  86. return await super.run(data, Object.assign({}, opts, {imageFiles}));
  87. }
  88. }
  89. module.exports = ConvertDjvu;