mirror of
https://github.com/Stirling-Tools/Stirling-PDF.git
synced 2025-05-21 17:32:01 +00:00

# Description of Changes - **What was changed** - Updated controller methods to use strongly‐typed primitives (`int`, `long`, `boolean`) instead of `String` for numeric and boolean parameters, eliminating calls to `Integer.parseInt`/`Long.parseLong` and improving null‐safety (`Boolean.TRUE.equals(...)`). - Enhanced all API request model classes with richer Swagger/OpenAPI annotations: added `requiredMode`, `defaultValue`, `allowableValues`, `format`, `pattern`, and tightened schema descriptions for all fields. - Refactored HTML form templates for “Remove Blank Pages” to include `min`, `max`, and `step` attributes on numeric inputs, matching the updated validation rules. - **Why the change was made** - **Type safety & robustness**: Shifting from `String` to native types prevents runtime parsing errors, simplifies controller logic, and makes default values explicit. - **Better API documentation & validation**: Enriching the Swagger annotations ensures generated docs accurately reflect required fields, default values, and permitted ranges, which improves client code generation and developer experience. - **Consistency across codebase**: Aligning all request models and controllers enforces a uniform coding style and reduces bugs. #3406 --- ## Checklist ### General - [x] I have read the [Contribution Guidelines](https://github.com/Stirling-Tools/Stirling-PDF/blob/main/CONTRIBUTING.md) - [x] I have read the [Stirling-PDF Developer Guide](https://github.com/Stirling-Tools/Stirling-PDF/blob/main/DeveloperGuide.md) (if applicable) - [ ] I have read the [How to add new languages to Stirling-PDF](https://github.com/Stirling-Tools/Stirling-PDF/blob/main/HowToAddNewLanguage.md) (if applicable) - [x] I have performed a self-review of my own code - [x] My changes generate no new warnings ### Documentation - [ ] I have updated relevant docs on [Stirling-PDF's doc repo](https://github.com/Stirling-Tools/Stirling-Tools.github.io/blob/main/docs/) (if functionality has heavily changed) - [ ] I have read the section [Add New Translation Tags](https://github.com/Stirling-Tools/Stirling-PDF/blob/main/HowToAddNewLanguage.md#add-new-translation-tags) (for new translation tags only) ### UI Changes (if applicable) - [ ] Screenshots or videos demonstrating the UI changes are attached (e.g., as comments or direct attachments in the PR) ### Testing (if applicable) - [ ] I have tested my changes locally. Refer to the [Testing Guide](https://github.com/Stirling-Tools/Stirling-PDF/blob/main/DeveloperGuide.md#6-testing) for more details.
201 lines
8.7 KiB
Java
201 lines
8.7 KiB
Java
package stirling.software.SPDF.controller.api;
|
|
|
|
import java.io.ByteArrayOutputStream;
|
|
import java.io.File;
|
|
import java.io.IOException;
|
|
import java.nio.file.Files;
|
|
import java.nio.file.Paths;
|
|
import java.nio.file.attribute.BasicFileAttributes;
|
|
import java.util.ArrayList;
|
|
import java.util.Arrays;
|
|
import java.util.Comparator;
|
|
import java.util.List;
|
|
|
|
import org.apache.pdfbox.multipdf.PDFMergerUtility;
|
|
import org.apache.pdfbox.pdmodel.PDDocument;
|
|
import org.apache.pdfbox.pdmodel.PDDocumentCatalog;
|
|
import org.apache.pdfbox.pdmodel.PDPage;
|
|
import org.apache.pdfbox.pdmodel.interactive.form.PDAcroForm;
|
|
import org.apache.pdfbox.pdmodel.interactive.form.PDField;
|
|
import org.apache.pdfbox.pdmodel.interactive.form.PDSignatureField;
|
|
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 lombok.RequiredArgsConstructor;
|
|
import lombok.extern.slf4j.Slf4j;
|
|
|
|
import stirling.software.SPDF.model.api.general.MergePdfsRequest;
|
|
import stirling.software.SPDF.service.CustomPDFDocumentFactory;
|
|
import stirling.software.SPDF.utils.GeneralUtils;
|
|
import stirling.software.SPDF.utils.WebResponseUtils;
|
|
|
|
@RestController
|
|
@Slf4j
|
|
@RequestMapping("/api/v1/general")
|
|
@Tag(name = "General", description = "General APIs")
|
|
@RequiredArgsConstructor
|
|
public class MergeController {
|
|
|
|
private final CustomPDFDocumentFactory pdfDocumentFactory;
|
|
|
|
// Merges a list of PDDocument objects into a single PDDocument
|
|
public PDDocument mergeDocuments(List<PDDocument> documents) throws IOException {
|
|
PDDocument mergedDoc = pdfDocumentFactory.createNewDocument();
|
|
for (PDDocument doc : documents) {
|
|
for (PDPage page : doc.getPages()) {
|
|
mergedDoc.addPage(page);
|
|
}
|
|
}
|
|
return mergedDoc;
|
|
}
|
|
|
|
// Returns a comparator for sorting MultipartFile arrays based on the given sort type
|
|
private Comparator<MultipartFile> getSortComparator(String sortType) {
|
|
switch (sortType) {
|
|
case "byFileName":
|
|
return Comparator.comparing(MultipartFile::getOriginalFilename);
|
|
case "byDateModified":
|
|
return (file1, file2) -> {
|
|
try {
|
|
BasicFileAttributes attr1 =
|
|
Files.readAttributes(
|
|
Paths.get(file1.getOriginalFilename()),
|
|
BasicFileAttributes.class);
|
|
BasicFileAttributes attr2 =
|
|
Files.readAttributes(
|
|
Paths.get(file2.getOriginalFilename()),
|
|
BasicFileAttributes.class);
|
|
return attr1.lastModifiedTime().compareTo(attr2.lastModifiedTime());
|
|
} catch (IOException e) {
|
|
return 0; // If there's an error, treat them as equal
|
|
}
|
|
};
|
|
case "byDateCreated":
|
|
return (file1, file2) -> {
|
|
try {
|
|
BasicFileAttributes attr1 =
|
|
Files.readAttributes(
|
|
Paths.get(file1.getOriginalFilename()),
|
|
BasicFileAttributes.class);
|
|
BasicFileAttributes attr2 =
|
|
Files.readAttributes(
|
|
Paths.get(file2.getOriginalFilename()),
|
|
BasicFileAttributes.class);
|
|
return attr1.creationTime().compareTo(attr2.creationTime());
|
|
} catch (IOException e) {
|
|
return 0; // If there's an error, treat them as equal
|
|
}
|
|
};
|
|
case "byPDFTitle":
|
|
return (file1, file2) -> {
|
|
try (PDDocument doc1 = pdfDocumentFactory.load(file1);
|
|
PDDocument doc2 = pdfDocumentFactory.load(file2)) {
|
|
String title1 = doc1.getDocumentInformation().getTitle();
|
|
String title2 = doc2.getDocumentInformation().getTitle();
|
|
return title1.compareTo(title2);
|
|
} catch (IOException e) {
|
|
return 0;
|
|
}
|
|
};
|
|
case "orderProvided":
|
|
default:
|
|
return (file1, file2) -> 0; // Default is the order provided
|
|
}
|
|
}
|
|
|
|
@PostMapping(consumes = "multipart/form-data", value = "/merge-pdfs")
|
|
@Operation(
|
|
summary = "Merge multiple PDF files into one",
|
|
description =
|
|
"This endpoint merges multiple PDF files into a single PDF file. The merged"
|
|
+ " file will contain all pages from the input files in the order they were"
|
|
+ " provided. Input:PDF Output:PDF Type:MISO")
|
|
public ResponseEntity<byte[]> mergePdfs(@ModelAttribute MergePdfsRequest request)
|
|
throws IOException {
|
|
List<File> filesToDelete = new ArrayList<>(); // List of temporary files to delete
|
|
File mergedTempFile = null;
|
|
PDDocument mergedDocument = null;
|
|
|
|
boolean removeCertSign = Boolean.TRUE.equals(request.getRemoveCertSign());
|
|
|
|
try {
|
|
MultipartFile[] files = request.getFileInput();
|
|
Arrays.sort(
|
|
files,
|
|
getSortComparator(
|
|
request.getSortType())); // Sort files based on the given sort type
|
|
|
|
PDFMergerUtility mergerUtility = new PDFMergerUtility();
|
|
long totalSize = 0;
|
|
for (MultipartFile multipartFile : files) {
|
|
totalSize += multipartFile.getSize();
|
|
File tempFile =
|
|
GeneralUtils.convertMultipartFileToFile(
|
|
multipartFile); // Convert MultipartFile to File
|
|
filesToDelete.add(tempFile); // Add temp file to the list for later deletion
|
|
mergerUtility.addSource(tempFile); // Add source file to the merger utility
|
|
}
|
|
|
|
mergedTempFile = Files.createTempFile("merged-", ".pdf").toFile();
|
|
mergerUtility.setDestinationFileName(mergedTempFile.getAbsolutePath());
|
|
|
|
mergerUtility.mergeDocuments(
|
|
pdfDocumentFactory.getStreamCacheFunction(totalSize)); // Merge the documents
|
|
|
|
// Load the merged PDF document
|
|
mergedDocument = pdfDocumentFactory.load(mergedTempFile);
|
|
|
|
// Remove signatures if removeCertSign is true
|
|
if (removeCertSign) {
|
|
PDDocumentCatalog catalog = mergedDocument.getDocumentCatalog();
|
|
PDAcroForm acroForm = catalog.getAcroForm();
|
|
if (acroForm != null) {
|
|
List<PDField> fieldsToRemove =
|
|
acroForm.getFields().stream()
|
|
.filter(field -> field instanceof PDSignatureField)
|
|
.toList();
|
|
|
|
if (!fieldsToRemove.isEmpty()) {
|
|
acroForm.flatten(
|
|
fieldsToRemove,
|
|
false); // Flatten the fields, effectively removing them
|
|
}
|
|
}
|
|
}
|
|
|
|
// Save the modified document to a new ByteArrayOutputStream
|
|
ByteArrayOutputStream baos = new ByteArrayOutputStream();
|
|
mergedDocument.save(baos);
|
|
|
|
String mergedFileName =
|
|
files[0].getOriginalFilename().replaceFirst("[.][^.]+$", "")
|
|
+ "_merged_unsigned.pdf";
|
|
return WebResponseUtils.boasToWebResponse(
|
|
baos, mergedFileName); // Return the modified PDF
|
|
|
|
} catch (Exception ex) {
|
|
log.error("Error in merge pdf process", ex);
|
|
throw ex;
|
|
} finally {
|
|
if (mergedDocument != null) {
|
|
mergedDocument.close(); // Close the merged document
|
|
}
|
|
for (File file : filesToDelete) {
|
|
if (file != null) {
|
|
Files.deleteIfExists(file.toPath()); // Delete temporary files
|
|
}
|
|
}
|
|
if (mergedTempFile != null) {
|
|
Files.deleteIfExists(mergedTempFile.toPath());
|
|
}
|
|
}
|
|
}
|
|
}
|