All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.jodconverter.local.office.LocalOfficeManagerPoolEntry Maven / Gradle / Ivy

Go to download

Module required in order to process local conversions for the Java OpenDocument Converter (JODConverter) project.

The newest version!
/*
 * Copyright (c) 2004 - 2012; Mirko Nasato and contributors
 *               2016 - 2022; Simon Braconnier and contributors
 *               2022 - present; JODConverter
 *
 * This file is part of JODConverter - Java OpenDocument Converter.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package org.jodconverter.local.office;

import java.util.concurrent.TimeoutException;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicInteger;

import org.checkerframework.checker.nullness.qual.NonNull;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import org.jodconverter.core.office.AbstractOfficeManagerPoolEntry;
import org.jodconverter.core.office.OfficeException;
import org.jodconverter.core.task.OfficeTask;

/**
 * An {@link LocalOfficeManagerPoolEntry} is responsible to execute tasks submitted through a {@link
 * org.jodconverter.local.office.LocalOfficeManager}. It will submit tasks to its inner {@link
 * LocalOfficeProcessManager} and wait until the task is done or a configured task execution timeout
 * is reached.
 *
 * 

An {@link LocalOfficeManagerPoolEntry} is also responsible to restart an office process when * the maximum number of tasks per process is reached. * * @see org.jodconverter.local.office.LocalOfficeManager * @see LocalOfficeProcessManager */ class LocalOfficeManagerPoolEntry extends AbstractOfficeManagerPoolEntry { private static final Logger LOGGER = LoggerFactory.getLogger(LocalOfficeManagerPoolEntry.class); private final int maxTasksPerProcess; private final LocalOfficeProcessManager officeProcessManager; private final AtomicInteger taskCount = new AtomicInteger(0); private final AtomicBoolean disconnectExpected = new AtomicBoolean(false); /** * Creates a new pool entry for the specified office URL with the specified configuration. * * @param maxTasksPerProcess The maximum number of tasks an office process can execute before * restarting. * @param taskExecutionTimeout The maximum time allowed to process a task. If the processing time * of a task is longer than this timeout, this task will be aborted and the next task is * processed. * @param officeProcessManager The office process manager. */ /* default */ LocalOfficeManagerPoolEntry( final int maxTasksPerProcess, final long taskExecutionTimeout, final LocalOfficeProcessManager officeProcessManager) { super(taskExecutionTimeout); this.officeProcessManager = officeProcessManager; this.maxTasksPerProcess = maxTasksPerProcess; // This connection event listener will be notified when a connection is established or // closed/lost to/from an office instance. final OfficeConnectionEventListener connectionEventListener = new OfficeConnectionEventListener() { // A connection is established. @Override public void connected(final OfficeConnectionEvent event) { // Reset the task count and make the manager available. taskCount.set(0); setAvailable(true); } // A connection is closed/lost. @Override public void disconnected(final OfficeConnectionEvent event) { // Make the manager unavailable. setAvailable(false); // When it comes from an expected behavior (we have put // the field to true before calling a function), just reset // the disconnectExpected value to false. When we didn't expect // the disconnection, we must restart the office process, which // will cancel any task that may be running. if (!disconnectExpected.compareAndSet(true, false)) { // Here, we didn't expect this disconnection. We must restart // the office process, canceling any task that may be running. LOGGER.warn("Connection lost unexpectedly; attempting restart"); cancelTask(); officeProcessManager.restartDueToLostConnection(); } } }; // Listen to any connection events to the office instance. officeProcessManager.getConnection().addConnectionEventListener(connectionEventListener); } @Override public void doExecute(final @NonNull OfficeTask task) throws OfficeException { LOGGER.debug("Executing task: {}", task); // Execute the task. task.execute(officeProcessManager.getConnection()); LOGGER.debug("Task executed successfully: {}", task); // Increment the task count final int count = taskCount.incrementAndGet(); // Now check if the office process must be restarted. if (count == maxTasksPerProcess) { LOGGER.info( "Reached limit of {} maximum tasks per process; restarting...", maxTasksPerProcess); restart(); } else { LOGGER.debug( "Limit of {} maximum tasks per process not reached yet. Task count is {}", maxTasksPerProcess, count); } } @Override protected void handleExecuteTimeoutException(final @NonNull TimeoutException timeoutEx) { // Is the task did not complete within the configured timeout, we must restart officeProcessManager.restartDueToTaskTimeout(); } @Override public boolean isRunning() { return super.isRunning() && officeProcessManager.getConnection().isConnected(); } @Override public void doStart() throws OfficeException { // Start the office process and connect to it. officeProcessManager.start(); } @Override public void doStop() throws OfficeException { // The manager is no longer available setAvailable(false); // From here on, any disconnection from an office process is expected. disconnectExpected.set(true); // Now we can stop the running office process officeProcessManager.stop(); } private void restart() { // The manager is no longer available setAvailable(false); // Indicates that the disconnection to follow is expected disconnectExpected.set(true); // Restart the office instance officeProcessManager.restart(); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy