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

org.jodconverter.office.OfficeProcessManagerPoolEntry Maven / Gradle / Ivy

Go to download

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

There is a newer version: 4.4.8
Show newest version
/*
 * Copyright 2004 - 2012 Mirko Nasato and contributors
 *           2016 - 2020 Simon Braconnier and contributors
 *
 * 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.office;

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

import com.sun.star.beans.PropertyValue;
import com.sun.star.beans.XHierarchicalPropertySet;
import com.sun.star.beans.XHierarchicalPropertySetInfo;
import com.sun.star.lang.XComponent;
import com.sun.star.lang.XMultiServiceFactory;
import com.sun.star.uno.XComponentContext;
import com.sun.star.util.XChangesBatch;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import org.jodconverter.office.utils.Lo;
import org.jodconverter.task.OfficeTask;

/**
 * A OfficeProcessManagerPoolEntry is responsible to execute tasks submitted through a {@link
 * LocalOfficeManager}. It will submit tasks to its inner {@link OfficeProcessManager} and wait
 * until the task is done or a configured task execution timeout is reached.
 *
 * 

A OfficeProcessManagerPoolEntry is also responsible to restart an office process when the * maximum number of tasks per process is reached. * * @see OfficeProcessManager * @see LocalOfficeManager */ class OfficeProcessManagerPoolEntry extends AbstractOfficeManagerPoolEntry { private static final String PROPPATH_USE_OPENGL = "VCL/UseOpenGL"; private static final Logger LOGGER = LoggerFactory.getLogger(OfficeProcessManagerPoolEntry.class); private final OfficeProcessManager 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 default configuration. * * @param officeUrl The URL for which the entry is created. */ public OfficeProcessManagerPoolEntry(final OfficeUrl officeUrl) { this(officeUrl, new OfficeProcessManagerPoolEntryConfig()); } /** * Creates a new pool entry for the specified office URL with the specified configuration. * * @param officeUrl The URL for which the entry is created. * @param config The entry configuration. */ public OfficeProcessManagerPoolEntry( final OfficeUrl officeUrl, final OfficeProcessManagerPoolEntryConfig config) { super(config); // Create the process manager that will deal with the office instance officeProcessManager = new OfficeProcessManager(officeUrl, config); // 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 task executor available. taskCount.set(0); taskExecutor.setAvailable(true); } // A connection is closed/lost. @Override public void disconnected(final OfficeConnectionEvent event) { // Make the task executor unavailable. taskExecutor.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"); if (currentFuture != null) { currentFuture.cancel(true); } officeProcessManager.restartDueToLostConnection(); } } }; // Listen to any connection events to the office instance. officeProcessManager.getConnection().addConnectionEventListener(connectionEventListener); } @Override public void doExecute(final OfficeTask task) throws OfficeException { final OfficeProcessManagerPoolEntryConfig entryConfig = (OfficeProcessManagerPoolEntryConfig) config; // First check if the office process must be restarted final int count = taskCount.getAndIncrement(); if (entryConfig.getMaxTasksPerProcess() > 0 && count == entryConfig.getMaxTasksPerProcess()) { LOGGER.info( "Reached limit of {} maximum tasks per process; restarting...", entryConfig.getMaxTasksPerProcess()); restart(); // taskCount will be 0 rather than 1 at this point, so fix this. taskCount.getAndIncrement(); } // Execute the task task.execute(officeProcessManager.getConnection()); } @Override protected void handleExecuteTimeoutException(final TimeoutException timeoutEx) { // Is the 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.startAndWait(); // Here a connection has been made successfully. Check to disable // the usage of OpenGL. Some file won't load properly if OpenGL // is on (LibreOffice). final OfficeProcessManagerPoolEntryConfig entryConfig = (OfficeProcessManagerPoolEntryConfig) config; if (entryConfig.isDisableOpengl() && disableOpengl(officeProcessManager.getConnection().getComponentContext())) { LOGGER.info("OpenGL has been disabled and a restart is required; restarting..."); restart(); } } @Override public void doStop() throws OfficeException { // From here on, any disconnection from an office process is expected. disconnectExpected.set(true); // Now we can stopped the running office process officeProcessManager.stopAndWait(); } private void restart() throws OfficeException { // The executor is no longer available taskExecutor.setAvailable(false); // Indicates that the disconnection to follow is expected disconnectExpected.set(true); // Restart the office instance officeProcessManager.restartAndWait(); } // Create a configuration view for the specified configuration path. private Object createConfigurationView(final XMultiServiceFactory provider, final String path) throws com.sun.star.uno.Exception { // Creation arguments: nodepath final PropertyValue argument = new PropertyValue(); argument.Name = "nodepath"; argument.Value = path; final Object[] arguments = new Object[1]; arguments[0] = argument; // create the view return provider.createInstanceWithArguments( "com.sun.star.configuration.ConfigurationUpdateAccess", arguments); } private boolean disableOpengl(final XComponentContext officeContext) throws OfficeException { // See configuration registry for more options. // e.g: C:\Program Files\LibreOffice 5\share\registry\main.xcd try { // Create the view to the root element where UseOpenGL option lives final Object viewRoot = createConfigurationView( Lo.createInstanceMCF( officeContext, XMultiServiceFactory.class, "com.sun.star.configuration.ConfigurationProvider"), "/org.openoffice.Office.Common"); try { // Check if the OpenGL option is on final XHierarchicalPropertySet properties = Lo.qi(XHierarchicalPropertySet.class, viewRoot); final XHierarchicalPropertySetInfo propsInfo = properties.getHierarchicalPropertySetInfo(); if (propsInfo.hasPropertyByHierarchicalName(PROPPATH_USE_OPENGL)) { final boolean useOpengl = (boolean) properties.getHierarchicalPropertyValue(PROPPATH_USE_OPENGL); LOGGER.info("Use OpenGL is set to {}", useOpengl); if (useOpengl) { properties.setHierarchicalPropertyValue(PROPPATH_USE_OPENGL, false); // Changes have been applied to the view here final XChangesBatch updateControl = Lo.qi(XChangesBatch.class, viewRoot); updateControl.commitChanges(); // A restart is required. return true; } } } finally { // We are done with the view - dispose it Lo.qi(XComponent.class, viewRoot).dispose(); } return false; // No restart needed } catch (com.sun.star.uno.Exception ex) { throw new OfficeException("Unable to check if the Use OpenGL option is on.", ex); } } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy