diff --git a/build.gradle b/build.gradle index f506ef019b3..40dda97f2e6 100644 --- a/build.gradle +++ b/build.gradle @@ -12,7 +12,7 @@ plugins { import com.github.jk1.license.render.* group = 'stirling.software' -version = '0.19.0' +version = '0.19.1' sourceCompatibility = '17' repositories { diff --git a/src/main/java/stirling/software/SPDF/config/AppConfig.java b/src/main/java/stirling/software/SPDF/config/AppConfig.java index 0411715f51c..cf05fd1b0c6 100644 --- a/src/main/java/stirling/software/SPDF/config/AppConfig.java +++ b/src/main/java/stirling/software/SPDF/config/AppConfig.java @@ -1,5 +1,8 @@ package stirling.software.SPDF.config; +import java.nio.file.Files; +import java.nio.file.Paths; + import org.springframework.beans.factory.annotation.Autowired; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; @@ -57,4 +60,22 @@ public boolean rateLimit() { if (appName == null) appName = System.getenv("rateLimit"); return (appName != null) ? Boolean.valueOf(appName) : false; } + + @Bean(name = "RunningInDocker") + public boolean runningInDocker() { + return Files.exists(Paths.get("/.dockerenv")); + } + + @Bean(name = "bookFormatsInstalled") + public boolean bookFormatsInstalled() { + return applicationProperties.getSystem().getCustomApplications().isInstallBookFormats(); + } + + @Bean(name = "htmlFormatsInstalled") + public boolean htmlFormatsInstalled() { + return applicationProperties + .getSystem() + .getCustomApplications() + .isInstallAdvancedHtmlToPDF(); + } } diff --git a/src/main/java/stirling/software/SPDF/config/EndpointConfiguration.java b/src/main/java/stirling/software/SPDF/config/EndpointConfiguration.java index 593b70b43db..f1e328f94b8 100644 --- a/src/main/java/stirling/software/SPDF/config/EndpointConfiguration.java +++ b/src/main/java/stirling/software/SPDF/config/EndpointConfiguration.java @@ -9,11 +9,14 @@ import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.beans.factory.annotation.Qualifier; +import org.springframework.context.annotation.DependsOn; import org.springframework.stereotype.Service; import stirling.software.SPDF.model.ApplicationProperties; @Service +@DependsOn({"bookFormatsInstalled"}) public class EndpointConfiguration { private static final Logger logger = LoggerFactory.getLogger(EndpointConfiguration.class); private Map endpointStatuses = new ConcurrentHashMap<>(); @@ -21,9 +24,14 @@ public class EndpointConfiguration { private final ApplicationProperties applicationProperties; + private boolean bookFormatsInstalled; + @Autowired - public EndpointConfiguration(ApplicationProperties applicationProperties) { + public EndpointConfiguration( + ApplicationProperties applicationProperties, + @Qualifier("bookFormatsInstalled") boolean bookFormatsInstalled) { this.applicationProperties = applicationProperties; + this.bookFormatsInstalled = bookFormatsInstalled; init(); processEnvironmentConfigs(); } @@ -145,6 +153,12 @@ public void init() { addEndpointToGroup("CLI", "ocr-pdf"); addEndpointToGroup("CLI", "html-to-pdf"); addEndpointToGroup("CLI", "url-to-pdf"); + addEndpointToGroup("CLI", "book-to-pdf"); + addEndpointToGroup("CLI", "pdf-to-book"); + + // Calibre + addEndpointToGroup("Calibre", "book-to-pdf"); + addEndpointToGroup("Calibre", "pdf-to-book"); // python addEndpointToGroup("Python", "extract-image-scans"); @@ -215,7 +229,9 @@ public void init() { private void processEnvironmentConfigs() { List endpointsToRemove = applicationProperties.getEndpoints().getToRemove(); List groupsToRemove = applicationProperties.getEndpoints().getGroupsToRemove(); - + if (!bookFormatsInstalled) { + groupsToRemove.add("Calibre"); + } if (endpointsToRemove != null) { for (String endpoint : endpointsToRemove) { disableEndpoint(endpoint.trim()); diff --git a/src/main/java/stirling/software/SPDF/config/PostStartupProcesses.java b/src/main/java/stirling/software/SPDF/config/PostStartupProcesses.java new file mode 100644 index 00000000000..862e5f9e521 --- /dev/null +++ b/src/main/java/stirling/software/SPDF/config/PostStartupProcesses.java @@ -0,0 +1,98 @@ +package stirling.software.SPDF.config; + +import java.io.IOException; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.List; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.beans.factory.annotation.Qualifier; +import org.springframework.stereotype.Component; + +import jakarta.annotation.PostConstruct; +import stirling.software.SPDF.model.ApplicationProperties; +import stirling.software.SPDF.utils.ProcessExecutor; +import stirling.software.SPDF.utils.ProcessExecutor.ProcessExecutorResult; + +@Component +public class PostStartupProcesses { + + @Autowired ApplicationProperties applicationProperties; + + @Autowired + @Qualifier("RunningInDocker") + private boolean runningInDocker; + + @Autowired + @Qualifier("bookFormatsInstalled") + private boolean bookFormatsInstalled; + + @Autowired + @Qualifier("htmlFormatsInstalled") + private boolean htmlFormatsInstalled; + + private static final Logger logger = LoggerFactory.getLogger(PostStartupProcesses.class); + + @PostConstruct + public void runInstallCommandBasedOnEnvironment() throws IOException, InterruptedException { + List> commands = new ArrayList<>(); + // Checking for DOCKER_INSTALL_BOOK_FORMATS environment variable + if (bookFormatsInstalled) { + List tmpList = new ArrayList<>(); + // Set up the timezone configuration commands + tmpList.addAll( + Arrays.asList( + "sh", + "-c", + "echo 'tzdata tzdata/Areas select Europe' | debconf-set-selections; " + + "echo 'tzdata tzdata/Zones/Europe select Berlin' | debconf-set-selections")); + commands.add(tmpList); + + // Install calibre with DEBIAN_FRONTEND set to noninteractive + tmpList = new ArrayList<>(); + tmpList.addAll( + Arrays.asList( + "sh", + "-c", + "DEBIAN_FRONTEND=noninteractive apt-get install -y --no-install-recommends calibre")); + commands.add(tmpList); + } + + // Checking for DOCKER_INSTALL_HTML_FORMATS environment variable + if (htmlFormatsInstalled) { + List tmpList = new ArrayList<>(); + // Add -y flag for automatic yes to prompts and --no-install-recommends to reduce size + tmpList.addAll( + Arrays.asList( + "apt-get", "install", "wkhtmltopdf", "-y", "--no-install-recommends")); + commands.add(tmpList); + } + + if (!commands.isEmpty()) { + // Run the command + if (runningInDocker) { + List tmpList = new ArrayList<>(); + tmpList.addAll(Arrays.asList("apt-get", "update")); + commands.add(0, tmpList); + + for (List list : commands) { + ProcessExecutorResult returnCode = + ProcessExecutor.getInstance(ProcessExecutor.Processes.INSTALL_APP, true) + .runCommandWithOutputHandling(list); + logger.info("RC for app installs {}", returnCode.getRc()); + } + } else { + + logger.info( + "Not running inside Docker so skipping automated install process with command."); + } + + } else { + if (runningInDocker) { + logger.info("No custom apps to install."); + } + } + } +} diff --git a/src/main/java/stirling/software/SPDF/config/security/UserAuthenticationFilter.java b/src/main/java/stirling/software/SPDF/config/security/UserAuthenticationFilter.java index 47423eb6724..61b209de685 100644 --- a/src/main/java/stirling/software/SPDF/config/security/UserAuthenticationFilter.java +++ b/src/main/java/stirling/software/SPDF/config/security/UserAuthenticationFilter.java @@ -115,4 +115,4 @@ protected boolean shouldNotFilter(HttpServletRequest request) throws ServletExce return false; } -} +} \ No newline at end of file diff --git a/src/main/java/stirling/software/SPDF/controller/api/converters/ConvertBookToPDFController.java b/src/main/java/stirling/software/SPDF/controller/api/converters/ConvertBookToPDFController.java new file mode 100644 index 00000000000..453f8e6e8ef --- /dev/null +++ b/src/main/java/stirling/software/SPDF/controller/api/converters/ConvertBookToPDFController.java @@ -0,0 +1,68 @@ +package stirling.software.SPDF.controller.api.converters; + +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.beans.factory.annotation.Qualifier; +import org.springframework.http.ResponseEntity; +import org.springframework.web.bind.annotation.ModelAttribute; +import org.springframework.web.bind.annotation.PostMapping; +import org.springframework.web.bind.annotation.RequestMapping; +import org.springframework.web.bind.annotation.RestController; +import org.springframework.web.multipart.MultipartFile; + +import io.swagger.v3.oas.annotations.Operation; +import io.swagger.v3.oas.annotations.tags.Tag; + +import stirling.software.SPDF.model.api.GeneralFile; +import stirling.software.SPDF.utils.FileToPdf; +import stirling.software.SPDF.utils.WebResponseUtils; + +@RestController +@Tag(name = "Convert", description = "Convert APIs") +@RequestMapping("/api/v1/convert") +public class ConvertBookToPDFController { + + @Autowired + @Qualifier("bookFormatsInstalled") + private boolean bookFormatsInstalled; + + @PostMapping(consumes = "multipart/form-data", value = "/book/pdf") + @Operation( + summary = + "Convert a BOOK/comic (*.epub | *.mobi | *.azw3 | *.fb2 | *.txt | *.docx) to PDF", + description = + "(Requires bookFormatsInstalled flag and Calibre installed) This endpoint takes an BOOK/comic (*.epub | *.mobi | *.azw3 | *.fb2 | *.txt | *.docx) input and converts it to PDF format.") + public ResponseEntity HtmlToPdf(@ModelAttribute GeneralFile request) throws Exception { + MultipartFile fileInput = request.getFileInput(); + + if (!bookFormatsInstalled) { + throw new IllegalArgumentException( + "bookFormatsInstalled flag is False, this functionality is not avaiable"); + } + + if (fileInput == null) { + throw new IllegalArgumentException("Please provide a file for conversion."); + } + + String originalFilename = fileInput.getOriginalFilename(); + + if (originalFilename != null) { + String originalFilenameLower = originalFilename.toLowerCase(); + if (!originalFilenameLower.endsWith(".epub") + && !originalFilenameLower.endsWith(".mobi") + && !originalFilenameLower.endsWith(".azw3") + && !originalFilenameLower.endsWith(".fb2") + && !originalFilenameLower.endsWith(".txt") + && !originalFilenameLower.endsWith(".docx")) { + throw new IllegalArgumentException( + "File must be in .epub, .mobi, .azw3, .fb2, .txt, or .docx format."); + } + } + byte[] pdfBytes = FileToPdf.convertBookTypeToPdf(fileInput.getBytes(), originalFilename); + + String outputFilename = + originalFilename.replaceFirst("[.][^.]+$", "") + + ".pdf"; // Remove file extension and append .pdf + + return WebResponseUtils.bytesToWebResponse(pdfBytes, outputFilename); + } +} diff --git a/src/main/java/stirling/software/SPDF/controller/api/converters/ConvertHtmlToPDF.java b/src/main/java/stirling/software/SPDF/controller/api/converters/ConvertHtmlToPDF.java index bec090404c2..fdcd114a3b3 100644 --- a/src/main/java/stirling/software/SPDF/controller/api/converters/ConvertHtmlToPDF.java +++ b/src/main/java/stirling/software/SPDF/controller/api/converters/ConvertHtmlToPDF.java @@ -1,5 +1,7 @@ package stirling.software.SPDF.controller.api.converters; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.beans.factory.annotation.Qualifier; import org.springframework.http.ResponseEntity; import org.springframework.web.bind.annotation.ModelAttribute; import org.springframework.web.bind.annotation.PostMapping; @@ -19,6 +21,10 @@ @RequestMapping("/api/v1/convert") public class ConvertHtmlToPDF { + @Autowired + @Qualifier("htmlFormatsInstalled") + private boolean htmlFormatsInstalled; + @PostMapping(consumes = "multipart/form-data", value = "/html/pdf") @Operation( summary = "Convert an HTML or ZIP (containing HTML and CSS) to PDF", @@ -37,7 +43,9 @@ public ResponseEntity HtmlToPdf(@ModelAttribute GeneralFile request) thr || (!originalFilename.endsWith(".html") && !originalFilename.endsWith(".zip"))) { throw new IllegalArgumentException("File must be either .html or .zip format."); } - byte[] pdfBytes = FileToPdf.convertHtmlToPdf(fileInput.getBytes(), originalFilename); + byte[] pdfBytes = + FileToPdf.convertHtmlToPdf( + fileInput.getBytes(), originalFilename, htmlFormatsInstalled); String outputFilename = originalFilename.replaceFirst("[.][^.]+$", "") diff --git a/src/main/java/stirling/software/SPDF/controller/api/converters/ConvertMarkdownToPdf.java b/src/main/java/stirling/software/SPDF/controller/api/converters/ConvertMarkdownToPdf.java index 8bdc504940d..d0fd632dcd2 100644 --- a/src/main/java/stirling/software/SPDF/controller/api/converters/ConvertMarkdownToPdf.java +++ b/src/main/java/stirling/software/SPDF/controller/api/converters/ConvertMarkdownToPdf.java @@ -3,6 +3,8 @@ import org.commonmark.node.Node; import org.commonmark.parser.Parser; import org.commonmark.renderer.html.HtmlRenderer; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.beans.factory.annotation.Qualifier; import org.springframework.http.ResponseEntity; import org.springframework.web.bind.annotation.ModelAttribute; import org.springframework.web.bind.annotation.PostMapping; @@ -22,6 +24,10 @@ @RequestMapping("/api/v1/convert") public class ConvertMarkdownToPdf { + @Autowired + @Qualifier("htmlFormatsInstalled") + private boolean htmlFormatsInstalled; + @PostMapping(consumes = "multipart/form-data", value = "/markdown/pdf") @Operation( summary = "Convert a Markdown file to PDF", @@ -46,7 +52,9 @@ public ResponseEntity markdownToPdf(@ModelAttribute GeneralFile request) HtmlRenderer renderer = HtmlRenderer.builder().build(); String htmlContent = renderer.render(document); - byte[] pdfBytes = FileToPdf.convertHtmlToPdf(htmlContent.getBytes(), "converted.html"); + byte[] pdfBytes = + FileToPdf.convertHtmlToPdf( + htmlContent.getBytes(), "converted.html", htmlFormatsInstalled); String outputFilename = originalFilename.replaceFirst("[.][^.]+$", "") diff --git a/src/main/java/stirling/software/SPDF/controller/api/converters/ConvertPDFToBookController.java b/src/main/java/stirling/software/SPDF/controller/api/converters/ConvertPDFToBookController.java new file mode 100644 index 00000000000..1ee09d9e7a0 --- /dev/null +++ b/src/main/java/stirling/software/SPDF/controller/api/converters/ConvertPDFToBookController.java @@ -0,0 +1,101 @@ +package stirling.software.SPDF.controller.api.converters; + +import java.nio.file.Files; +import java.nio.file.Path; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.List; + +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.beans.factory.annotation.Qualifier; +import org.springframework.http.ResponseEntity; +import org.springframework.web.bind.annotation.ModelAttribute; +import org.springframework.web.bind.annotation.PostMapping; +import org.springframework.web.bind.annotation.RequestMapping; +import org.springframework.web.bind.annotation.RestController; +import org.springframework.web.multipart.MultipartFile; + +import io.swagger.v3.oas.annotations.Operation; +import io.swagger.v3.oas.annotations.tags.Tag; + +import stirling.software.SPDF.model.api.converters.PdfToBookRequest; +import stirling.software.SPDF.utils.ProcessExecutor; +import stirling.software.SPDF.utils.ProcessExecutor.ProcessExecutorResult; +import stirling.software.SPDF.utils.WebResponseUtils; + +@RestController +@Tag(name = "Convert", description = "Convert APIs") +@RequestMapping("/api/v1/convert") +public class ConvertPDFToBookController { + + @Autowired + @Qualifier("bookFormatsInstalled") + private boolean bookFormatsInstalled; + + @PostMapping(consumes = "multipart/form-data", value = "/pdf/book") + @Operation( + summary = + "Convert a PDF to a Book/comic (*.epub | *.mobi | *.azw3 | *.fb2 | *.txt | *.docx .. (others to include by chatgpt) to PDF", + description = + "(Requires bookFormatsInstalled flag and Calibre installed) This endpoint Convert a PDF to a Book/comic (*.epub | *.mobi | *.azw3 | *.fb2 | *.txt | *.docx .. (others to include by chatgpt) to PDF") + public ResponseEntity HtmlToPdf(@ModelAttribute PdfToBookRequest request) + throws Exception { + MultipartFile fileInput = request.getFileInput(); + + if (!bookFormatsInstalled) { + throw new IllegalArgumentException( + "bookFormatsInstalled flag is False, this functionality is not avaiable"); + } + + if (fileInput == null) { + throw new IllegalArgumentException("Please provide a file for conversion."); + } + + // Validate the output format + String outputFormat = request.getOutputFormat().toLowerCase(); + List allowedFormats = + Arrays.asList( + "epub", "mobi", "azw3", "docx", "rtf", "txt", "html", "lit", "fb2", "pdb", + "lrf"); + if (!allowedFormats.contains(outputFormat)) { + throw new IllegalArgumentException("Invalid output format: " + outputFormat); + } + + byte[] outputFileBytes; + List command = new ArrayList<>(); + Path tempOutputFile = + Files.createTempFile( + "output_", + "." + outputFormat); // Use the output format for the file extension + Path tempInputFile = null; + + try { + // Create temp input file from the provided PDF + tempInputFile = Files.createTempFile("input_", ".pdf"); // Assuming input is always PDF + Files.write(tempInputFile, fileInput.getBytes()); + + command.add("ebook-convert"); + command.add(tempInputFile.toString()); + command.add(tempOutputFile.toString()); + + ProcessExecutorResult returnCode = + ProcessExecutor.getInstance(ProcessExecutor.Processes.CALIBRE) + .runCommandWithOutputHandling(command); + + outputFileBytes = Files.readAllBytes(tempOutputFile); + } finally { + // Clean up temporary files + if (tempInputFile != null) { + Files.deleteIfExists(tempInputFile); + } + Files.deleteIfExists(tempOutputFile); + } + + String outputFilename = + fileInput.getOriginalFilename().replaceFirst("[.][^.]+$", "") + + "." + + outputFormat; // Remove file extension and append .pdf + + return WebResponseUtils.bytesToWebResponse(outputFileBytes, outputFilename); + } +} diff --git a/src/main/java/stirling/software/SPDF/controller/api/converters/ConvertWebsiteToPDF.java b/src/main/java/stirling/software/SPDF/controller/api/converters/ConvertWebsiteToPDF.java index 815018e8714..a6cd439b15c 100644 --- a/src/main/java/stirling/software/SPDF/controller/api/converters/ConvertWebsiteToPDF.java +++ b/src/main/java/stirling/software/SPDF/controller/api/converters/ConvertWebsiteToPDF.java @@ -6,6 +6,8 @@ import java.util.ArrayList; import java.util.List; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.beans.factory.annotation.Qualifier; import org.springframework.http.ResponseEntity; import org.springframework.web.bind.annotation.ModelAttribute; import org.springframework.web.bind.annotation.PostMapping; @@ -26,6 +28,10 @@ @RequestMapping("/api/v1/convert") public class ConvertWebsiteToPDF { + @Autowired + @Qualifier("htmlFormatsInstalled") + private boolean htmlFormatsInstalled; + @PostMapping(consumes = "multipart/form-data", value = "/url/pdf") @Operation( summary = "Convert a URL to a PDF", @@ -47,7 +53,11 @@ public ResponseEntity urlToPdf(@ModelAttribute UrlToPdfRequest request) // Prepare the OCRmyPDF command List command = new ArrayList<>(); - command.add("weasyprint"); + if (!htmlFormatsInstalled) { + command.add("weasyprint"); + } else { + command.add("wkhtmltopdf"); + } command.add(URL); command.add(tempOutputFile.toString()); diff --git a/src/main/java/stirling/software/SPDF/controller/web/ConverterWebController.java b/src/main/java/stirling/software/SPDF/controller/web/ConverterWebController.java index 16e42decb1e..55ebcb914f3 100644 --- a/src/main/java/stirling/software/SPDF/controller/web/ConverterWebController.java +++ b/src/main/java/stirling/software/SPDF/controller/web/ConverterWebController.java @@ -1,5 +1,6 @@ package stirling.software.SPDF.controller.web; +import org.springframework.boot.autoconfigure.condition.ConditionalOnExpression; import org.springframework.stereotype.Controller; import org.springframework.ui.Model; import org.springframework.web.bind.annotation.GetMapping; @@ -12,6 +13,22 @@ @Tag(name = "Convert", description = "Convert APIs") public class ConverterWebController { + @ConditionalOnExpression("#{bookFormatsInstalled}") + @GetMapping("/book-to-pdf") + @Hidden + public String convertBookToPdfForm(Model model) { + model.addAttribute("currentPage", "book-to-pdf"); + return "convert/book-to-pdf"; + } + + @ConditionalOnExpression("#{bookFormatsInstalled}") + @GetMapping("/pdf-to-book") + @Hidden + public String convertPdfToBookForm(Model model) { + model.addAttribute("currentPage", "pdf-to-book"); + return "convert/pdf-to-book"; + } + @GetMapping("/img-to-pdf") @Hidden public String convertImgToPdfForm(Model model) { diff --git a/src/main/java/stirling/software/SPDF/controller/web/HomeWebController.java b/src/main/java/stirling/software/SPDF/controller/web/HomeWebController.java index 2426d5231a7..94e83342e39 100644 --- a/src/main/java/stirling/software/SPDF/controller/web/HomeWebController.java +++ b/src/main/java/stirling/software/SPDF/controller/web/HomeWebController.java @@ -1,7 +1,8 @@ package stirling.software.SPDF.controller.web; import java.io.IOException; -import java.nio.file.Files; +import java.io.InputStream; +import java.nio.charset.StandardCharsets; import java.util.List; import java.util.Map; @@ -38,7 +39,8 @@ public String licensesForm(Model model) { model.addAttribute("currentPage", "licenses"); Resource resource = new ClassPathResource("static/3rdPartyLicenses.json"); try { - String json = new String(Files.readAllBytes(resource.getFile().toPath())); + InputStream is = resource.getInputStream(); + String json = new String(is.readAllBytes(), StandardCharsets.UTF_8); ObjectMapper mapper = new ObjectMapper(); Map> data = mapper.readValue(json, new TypeReference>>() {}); diff --git a/src/main/java/stirling/software/SPDF/model/ApplicationProperties.java b/src/main/java/stirling/software/SPDF/model/ApplicationProperties.java index a41d641c44a..3258d8b1d4c 100644 --- a/src/main/java/stirling/software/SPDF/model/ApplicationProperties.java +++ b/src/main/java/stirling/software/SPDF/model/ApplicationProperties.java @@ -210,6 +210,7 @@ public static class System { private String rootURIPath; private String customStaticFilePath; private Integer maxFileSize; + private CustomApplications customApplications; private Boolean enableAlphaFunctionality; @@ -261,6 +262,14 @@ public void setMaxFileSize(Integer maxFileSize) { this.maxFileSize = maxFileSize; } + public CustomApplications getCustomApplications() { + return customApplications != null ? customApplications : new CustomApplications(); + } + + public void setCustomApplications(CustomApplications customApplications) { + this.customApplications = customApplications; + } + @Override public String toString() { return "System [defaultLocale=" @@ -273,10 +282,42 @@ public String toString() { + customStaticFilePath + ", maxFileSize=" + maxFileSize + + ", customApplications=" + + customApplications + ", enableAlphaFunctionality=" + enableAlphaFunctionality + "]"; } + + public static class CustomApplications { + private boolean installBookFormats; + private boolean installAdvancedHtmlToPDF; + + public boolean isInstallBookFormats() { + return installBookFormats; + } + + public void setInstallBookFormats(boolean installBookFormats) { + this.installBookFormats = installBookFormats; + } + + public boolean isInstallAdvancedHtmlToPDF() { + return installAdvancedHtmlToPDF; + } + + public void setInstallAdvancedHtmlToPDF(boolean installAdvancedHtmlToPDF) { + this.installAdvancedHtmlToPDF = installAdvancedHtmlToPDF; + } + + @Override + public String toString() { + return "CustomApplications [installBookFormats=" + + installBookFormats + + ", installAdvancedHtmlToPDF=" + + installAdvancedHtmlToPDF + + "]"; + } + } } public static class Ui { diff --git a/src/main/java/stirling/software/SPDF/model/api/converters/PdfToBookRequest.java b/src/main/java/stirling/software/SPDF/model/api/converters/PdfToBookRequest.java new file mode 100644 index 00000000000..b3454afb3ff --- /dev/null +++ b/src/main/java/stirling/software/SPDF/model/api/converters/PdfToBookRequest.java @@ -0,0 +1,19 @@ +package stirling.software.SPDF.model.api.converters; + +import io.swagger.v3.oas.annotations.media.Schema; + +import lombok.Data; +import lombok.EqualsAndHashCode; +import stirling.software.SPDF.model.api.PDFFile; + +@Data +@EqualsAndHashCode(callSuper = true) +public class PdfToBookRequest extends PDFFile { + + @Schema( + description = "The output Ebook format", + allowableValues = { + "epub", "mobi", "azw3", "docx", "rtf", "txt", "html", "lit", "fb2", "pdb", "lrf" + }) + private String outputFormat; +} diff --git a/src/main/java/stirling/software/SPDF/utils/FileToPdf.java b/src/main/java/stirling/software/SPDF/utils/FileToPdf.java index 5e6825dd661..ebdbf4fa901 100644 --- a/src/main/java/stirling/software/SPDF/utils/FileToPdf.java +++ b/src/main/java/stirling/software/SPDF/utils/FileToPdf.java @@ -14,7 +14,9 @@ import stirling.software.SPDF.utils.ProcessExecutor.ProcessExecutorResult; public class FileToPdf { - public static byte[] convertHtmlToPdf(byte[] fileBytes, String fileName) + + public static byte[] convertHtmlToPdf( + byte[] fileBytes, String fileName, boolean htmlFormatsInstalled) throws IOException, InterruptedException { Path tempOutputFile = Files.createTempFile("output_", ".pdf"); @@ -29,11 +31,22 @@ public static byte[] convertHtmlToPdf(byte[] fileBytes, String fileName) } List command = new ArrayList<>(); - command.add("weasyprint"); + if (!htmlFormatsInstalled) { + command.add("weasyprint"); + } else { + command.add("wkhtmltopdf"); + command.add("--enable-local-file-access"); + } + command.add(tempInputFile.toString()); command.add(tempOutputFile.toString()); ProcessExecutorResult returnCode; if (fileName.endsWith(".zip")) { + + if (htmlFormatsInstalled) { + // command.add(1, "--allow"); + // command.add(2, tempInputFile.getParent().toString()); + } returnCode = ProcessExecutor.getInstance(ProcessExecutor.Processes.WEASYPRINT) .runCommandWithOutputHandling( @@ -97,4 +110,38 @@ private static Path unzipAndGetMainHtml(byte[] fileBytes) throws IOException { return htmlFiles.get(0); } } + + public static byte[] convertBookTypeToPdf(byte[] bytes, String originalFilename) + throws IOException, InterruptedException { + if (originalFilename == null || originalFilename.lastIndexOf('.') == -1) { + throw new IllegalArgumentException("Invalid original filename."); + } + + String fileExtension = originalFilename.substring(originalFilename.lastIndexOf('.')); + List command = new ArrayList<>(); + Path tempOutputFile = Files.createTempFile("output_", ".pdf"); + Path tempInputFile = null; + + try { + // Create temp file with appropriate extension + tempInputFile = Files.createTempFile("input_", fileExtension); + Files.write(tempInputFile, bytes); + + command.add("ebook-convert"); + command.add(tempInputFile.toString()); + command.add(tempOutputFile.toString()); + + ProcessExecutorResult returnCode = + ProcessExecutor.getInstance(ProcessExecutor.Processes.CALIBRE) + .runCommandWithOutputHandling(command); + + return Files.readAllBytes(tempOutputFile); + } finally { + // Clean up temporary files + if (tempInputFile != null) { + Files.deleteIfExists(tempInputFile); + } + Files.deleteIfExists(tempOutputFile); + } + } } diff --git a/src/main/java/stirling/software/SPDF/utils/ProcessExecutor.java b/src/main/java/stirling/software/SPDF/utils/ProcessExecutor.java index 385f3b8071a..23311bde126 100644 --- a/src/main/java/stirling/software/SPDF/utils/ProcessExecutor.java +++ b/src/main/java/stirling/software/SPDF/utils/ProcessExecutor.java @@ -4,26 +4,39 @@ import java.io.File; import java.io.IOException; import java.io.InputStreamReader; +import java.io.InterruptedIOException; import java.nio.charset.StandardCharsets; import java.util.ArrayList; import java.util.List; import java.util.Map; import java.util.concurrent.ConcurrentHashMap; import java.util.concurrent.Semaphore; +import java.util.concurrent.TimeUnit; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; public class ProcessExecutor { + private static final Logger logger = LoggerFactory.getLogger(ProcessExecutor.class); + public enum Processes { LIBRE_OFFICE, OCR_MY_PDF, PYTHON_OPENCV, GHOSTSCRIPT, - WEASYPRINT + WEASYPRINT, + INSTALL_APP, + CALIBRE } private static final Map instances = new ConcurrentHashMap<>(); public static ProcessExecutor getInstance(Processes processType) { + return getInstance(processType, true); + } + + public static ProcessExecutor getInstance(Processes processType, boolean liveUpdates) { return instances.computeIfAbsent( processType, key -> { @@ -34,15 +47,32 @@ public static ProcessExecutor getInstance(Processes processType) { case PYTHON_OPENCV -> 8; case GHOSTSCRIPT -> 16; case WEASYPRINT -> 16; + case INSTALL_APP -> 1; + case CALIBRE -> 1; + }; + + long timeoutMinutes = + switch (key) { + case LIBRE_OFFICE -> 30; + case OCR_MY_PDF -> 30; + case PYTHON_OPENCV -> 30; + case GHOSTSCRIPT -> 5; + case WEASYPRINT -> 30; + case INSTALL_APP -> 60; + case CALIBRE -> 30; }; - return new ProcessExecutor(semaphoreLimit); + return new ProcessExecutor(semaphoreLimit, liveUpdates, timeoutMinutes); }); } private final Semaphore semaphore; + private final boolean liveUpdates; + private long timeoutDuration; - private ProcessExecutor(int semaphoreLimit) { + private ProcessExecutor(int semaphoreLimit, boolean liveUpdates, long timeout) { this.semaphore = new Semaphore(semaphoreLimit); + this.liveUpdates = liveUpdates; + this.timeoutDuration = timeout; } public ProcessExecutorResult runCommandWithOutputHandling(List command) @@ -52,12 +82,12 @@ public ProcessExecutorResult runCommandWithOutputHandling(List command) public ProcessExecutorResult runCommandWithOutputHandling( List command, File workingDirectory) throws IOException, InterruptedException { - int exitCode = 1; String messages = ""; + int exitCode = 1; semaphore.acquire(); try { - System.out.print("Running command: " + String.join(" ", command)); + logger.info("Running command: " + String.join(" ", command)); ProcessBuilder processBuilder = new ProcessBuilder(command); // Use the working directory if it's set @@ -81,7 +111,11 @@ public ProcessExecutorResult runCommandWithOutputHandling( String line; while ((line = errorReader.readLine()) != null) { errorLines.add(line); + if (liveUpdates) logger.info(line); } + } catch (InterruptedIOException e) { + logger.warn( + "Error reader thread was interrupted due to timeout."); } catch (IOException e) { e.printStackTrace(); } @@ -98,7 +132,11 @@ public ProcessExecutorResult runCommandWithOutputHandling( String line; while ((line = outputReader.readLine()) != null) { outputLines.add(line); + if (liveUpdates) logger.info(line); } + } catch (InterruptedIOException e) { + logger.warn( + "Error reader thread was interrupted due to timeout."); } catch (IOException e) { e.printStackTrace(); } @@ -108,29 +146,42 @@ public ProcessExecutorResult runCommandWithOutputHandling( outputReaderThread.start(); // Wait for the conversion process to complete - exitCode = process.waitFor(); - + boolean finished = process.waitFor(timeoutDuration, TimeUnit.MINUTES); + + if (!finished) { + // Terminate the process + process.destroy(); + // Interrupt the reader threads + errorReaderThread.interrupt(); + outputReaderThread.interrupt(); + throw new IOException("Process timeout exceeded."); + } + exitCode = process.exitValue(); // Wait for the reader threads to finish errorReaderThread.join(); outputReaderThread.join(); - if (outputLines.size() > 0) { - String outputMessage = String.join("\n", outputLines); - messages += outputMessage; - System.out.println("Command output:\n" + outputMessage); - } + if (!liveUpdates) { + if (outputLines.size() > 0) { + String outputMessage = String.join("\n", outputLines); + messages += outputMessage; + logger.info("Command output:\n" + outputMessage); + } - if (errorLines.size() > 0) { - String errorMessage = String.join("\n", errorLines); - messages += errorMessage; - System.out.println("Command error output:\n" + errorMessage); - if (exitCode != 0) { - throw new IOException( - "Command process failed with exit code " - + exitCode - + ". Error message: " - + errorMessage); + if (errorLines.size() > 0) { + String errorMessage = String.join("\n", errorLines); + messages += errorMessage; + logger.warn("Command error output:\n" + errorMessage); + if (exitCode != 0) { + throw new IOException( + "Command process failed with exit code " + + exitCode + + ". Error message: " + + errorMessage); + } } + } else if (exitCode != 0) { + throw new IOException("Command process failed with exit code " + exitCode); } } finally { semaphore.release(); diff --git a/src/main/resources/settings.yml.template b/src/main/resources/settings.yml.template index 52d5e4de71c..00c5998e794 100644 --- a/src/main/resources/settings.yml.template +++ b/src/main/resources/settings.yml.template @@ -9,10 +9,14 @@ security: loginResetTimeMinutes : 120 # lock account for 2 hours after x attempts system: + defaultLocale: 'en-US' # Set the default language (e.g. 'de-DE', 'fr-FR', etc) googlevisibility: false # 'true' to allow Google visibility (via robots.txt), 'false' to disallow enableAlphaFunctionality: false # Set to enable functionality which might need more testing before it fully goes live (This feature might make no changes) - + customApplications: + installBookFormats: false # Installs Calibre for book format conversion (For non docker it must be manually downloaded but will need to be true to show in UI) + installAdvancedHtmlToPDF: false # DO NOT USE EXTERNALLY, NOT SAFE! Install wkHtmlToPDF (For non docker it must be manually downloaded but will need to be true to show in UI) + #ui: # appName: exampleAppName # Application's visible name # homeDescription: I am a description # Short description or tagline shown on homepage. diff --git a/src/main/resources/static/images/book.svg b/src/main/resources/static/images/book.svg new file mode 100644 index 00000000000..302acf09e08 --- /dev/null +++ b/src/main/resources/static/images/book.svg @@ -0,0 +1,3 @@ + + + \ No newline at end of file diff --git a/src/main/resources/static/js/multitool/PdfContainer.js b/src/main/resources/static/js/multitool/PdfContainer.js index 4c7c84975cd..4a53c961571 100644 --- a/src/main/resources/static/js/multitool/PdfContainer.js +++ b/src/main/resources/static/js/multitool/PdfContainer.js @@ -209,20 +209,21 @@ class PdfContainer { async exportPdf() { const pdfDoc = await PDFLib.PDFDocument.create(); - for (var i=0; i + + + + + +
+
+
+

+
+
+
+

+
+
+
+ + +
+

+

+
+
+
+
+
+
+ + diff --git a/src/main/resources/templates/convert/pdf-to-book.html b/src/main/resources/templates/convert/pdf-to-book.html new file mode 100644 index 00000000000..a136c5cf7b6 --- /dev/null +++ b/src/main/resources/templates/convert/pdf-to-book.html @@ -0,0 +1,55 @@ + + + + + + + +
+
+
+

+
+
+
+

+
+
+ +
+ + +
+
+ + +
+

+

+
+
+
+
+
+
+ + diff --git a/src/main/resources/templates/fragments/navbar.html b/src/main/resources/templates/fragments/navbar.html index 8e5ce7322bb..b090b08c86a 100644 --- a/src/main/resources/templates/fragments/navbar.html +++ b/src/main/resources/templates/fragments/navbar.html @@ -77,6 +77,7 @@
+
@@ -86,6 +87,7 @@
+
diff --git a/src/main/resources/templates/home.html b/src/main/resources/templates/home.html index 946cdd9cefd..52f945ebe51 100644 --- a/src/main/resources/templates/home.html +++ b/src/main/resources/templates/home.html @@ -99,6 +99,8 @@

+
+