org.sejda.impl.sambox.PdfToMultipleImageTask Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of sejda-sambox Show documentation
Show all versions of sejda-sambox Show documentation
Package containing tasks implemented using sambox.
/*
* Copyright 2015 by Edi Weissmann ([email protected])
*
* This file is part of the Sejda source code
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as
* published by the Free Software Foundation, either version 3 of the
* License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program. If not, see .
*/
package org.sejda.impl.sambox;
import org.sejda.core.Sejda;
import org.sejda.core.support.io.MultipleOutputWriter;
import org.sejda.core.support.util.RuntimeUtils;
import org.sejda.impl.sambox.component.DefaultPdfSourceOpener;
import org.sejda.impl.sambox.component.PDDocumentHandler;
import org.sejda.model.exception.TaskException;
import org.sejda.model.input.PdfSource;
import org.sejda.model.input.PdfSourceOpener;
import org.sejda.model.parameter.image.AbstractPdfToMultipleImageParameters;
import org.sejda.model.task.TaskExecutionContext;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.awt.image.BufferedImage;
import java.io.File;
import java.util.Set;
import static java.util.Optional.ofNullable;
import static org.sejda.commons.util.IOUtils.closeQuietly;
import static org.sejda.core.notification.dsl.ApplicationEventsNotifier.notifyEvent;
import static org.sejda.core.support.io.OutputWriters.newMultipleOutputWriter;
import static org.sejda.core.support.io.model.FileOutput.file;
import static org.sejda.core.support.prefix.NameGenerator.nameGenerator;
import static org.sejda.core.support.prefix.model.NameGenerationRequest.nameRequest;
import static org.sejda.model.util.IOUtils.createTemporaryBuffer;
/**
* SAMBox implementation of a task which converts a pdf document to a collection of images, one image per page.
*
* @param
* the type of parameters.
* @author Andrea Vacondio
*
*/
public class PdfToMultipleImageTask extends BasePdfToImageTask {
private static final Logger LOG = LoggerFactory.getLogger(PdfToMultipleImageTask.class);
private MultipleOutputWriter outputWriter;
private PdfSourceOpener sourceOpener;
private PDDocumentHandler documentHandler = null;
@Override
public void before(T parameters, TaskExecutionContext executionContext) throws TaskException {
super.before(parameters, executionContext);
outputWriter = newMultipleOutputWriter(parameters.getExistingOutputPolicy(), executionContext);
sourceOpener = new DefaultPdfSourceOpener(executionContext);
}
@Override
public void execute(T parameters) throws TaskException {
int currentStep = 0;
int totalSteps = parameters.getSourceList().size();
for (PdfSource> source : parameters.getSourceList()) {
currentStep++;
try {
LOG.debug("Opening {}", source);
executionContext().notifiableTaskMetadata().setCurrentSource(source);
documentHandler = source.open(sourceOpener);
Set requestedPages = parameters.getPages(documentHandler.getNumberOfPages());
if (!requestedPages.isEmpty()) {
LOG.trace("Found {} pages to convert", totalSteps);
for (int currentPage : requestedPages) {
if (Boolean.getBoolean(Sejda.PERFORM_MEMORY_OPTIMIZATIONS_PROPERTY_NAME)) {
int percentageMemoryUsed = RuntimeUtils.getPercentageMemoryUsed();
if (percentageMemoryUsed > 60) {
LOG.debug("Closing and reopening source doc, memory usage reached: {}%",
percentageMemoryUsed);
closeQuietly(documentHandler);
documentHandler = source.open(sourceOpener);
}
}
File tmpFile = createTemporaryBuffer();
int fileNumber = executionContext().incrementAndGetOutputDocumentsCounter();
LOG.debug("Created output temporary buffer {} ", tmpFile);
try {
LOG.trace("Converting page {}", currentPage);
BufferedImage pageImage = documentHandler.renderImage(currentPage,
parameters.getResolutionInDpi(), parameters.getOutputImageColorType());
getWriter().openDestination(tmpFile, parameters);
getWriter().write(pageImage, parameters);
getWriter().closeDestination();
String outName = ofNullable(parameters.getSpecificResultFilename(fileNumber,
"." + parameters.getOutputImageType().getExtension())).orElseGet(
() -> nameGenerator(parameters.getOutputPrefix()).generate(
nameRequest(parameters.getOutputImageType().getExtension()).page(
currentPage).originalName(source.getName())
.fileNumber(fileNumber)));
outputWriter.addOutput(file(tmpFile).name(outName));
} catch (TaskException e) {
executionContext().assertTaskIsLenient(e);
notifyEvent(executionContext().notifiableTaskMetadata()).taskWarning(
String.format("Page %d was skipped, could not be converted", currentPage), e);
}
}
} else {
throw new TaskException("No pages converted");
}
} finally {
closeQuietly(documentHandler);
}
notifyEvent(executionContext().notifiableTaskMetadata()).stepsCompleted(currentStep).outOf(totalSteps);
}
executionContext().notifiableTaskMetadata().clearCurrentSource();
parameters.getOutput().accept(outputWriter);
LOG.debug("Documents converted to {} and saved to {}", parameters.getOutputImageType(), parameters.getOutput());
}
@Override
public void after() {
super.after();
closeQuietly(documentHandler);
}
}