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

org.camunda.bpm.application.AbstractProcessApplication Maven / Gradle / Ivy

There is a newer version: 7.23.0-alpha2
Show newest version
/*
 * Copyright Camunda Services GmbH and/or licensed to Camunda Services GmbH
 * under one or more contributor license agreements. See the NOTICE file
 * distributed with this work for additional information regarding copyright
 * ownership. Camunda licenses this file to you under the Apache License,
 * Version 2.0; 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.camunda.bpm.application;

import org.camunda.bpm.application.impl.DefaultElResolverLookup;
import org.camunda.bpm.application.impl.ProcessApplicationLogger;
import org.camunda.bpm.application.impl.ProcessApplicationScriptEnvironment;
import org.camunda.bpm.container.RuntimeContainerDelegate;
import org.camunda.bpm.engine.ProcessEngines;
import org.camunda.bpm.engine.delegate.ExecutionListener;
import org.camunda.bpm.engine.delegate.TaskListener;
import org.camunda.bpm.engine.impl.ProcessEngineLogger;
import org.camunda.bpm.engine.impl.context.Context;
import org.camunda.bpm.impl.juel.jakarta.el.BeanELResolver;
import org.camunda.bpm.impl.juel.jakarta.el.ELResolver;
import org.camunda.bpm.engine.impl.scripting.ExecutableScript;
import org.camunda.bpm.engine.impl.util.ClassLoaderUtil;
import org.camunda.bpm.engine.impl.variable.serializer.VariableSerializers;
import org.camunda.bpm.engine.repository.DeploymentBuilder;

import javax.script.ScriptEngine;
import java.util.Collections;
import java.util.List;
import java.util.Map;
import java.util.ServiceLoader;
import java.util.concurrent.Callable;


/**
 * @author Daniel Meyer
 */
public abstract class AbstractProcessApplication implements ProcessApplicationInterface {

  private static ProcessApplicationLogger LOG = ProcessEngineLogger.PROCESS_APPLICATION_LOGGER;

  protected ELResolver processApplicationElResolver;
  protected BeanELResolver processApplicationBeanElResolver;
  protected ProcessApplicationScriptEnvironment processApplicationScriptEnvironment;

  protected VariableSerializers variableSerializers;

  protected boolean isDeployed = false;

  protected String defaultDeployToEngineName = ProcessEngines.NAME_DEFAULT;

  // deployment /////////////////////////////////////////////////////

  public void deploy() {
    if (isDeployed) {
      LOG.alreadyDeployed();
    } else {
      try {
        ProcessApplicationReference reference = getReference();
        Context.setCurrentProcessApplication(reference);

        // deploy the application
        RuntimeContainerDelegate.INSTANCE.get().deployProcessApplication(this);
        isDeployed = true;

      } finally {
        Context.removeCurrentProcessApplication();

      }
    }
  }

  public void undeploy() {
    if (!isDeployed) {
      LOG.notDeployed();
    } else {
      // delegate stopping of the process application to the runtime container.
      RuntimeContainerDelegate.INSTANCE.get().undeployProcessApplication(this);
      isDeployed = false;
    }
  }

  public void createDeployment(String processArchiveName, DeploymentBuilder deploymentBuilder) {
    // default implementation does nothing
  }

  // Runtime ////////////////////////////////////////////

  public String getName() {
    Class processApplicationClass = getClass();
    String name = null;

    ProcessApplication annotation = processApplicationClass.getAnnotation(ProcessApplication.class);
    if (annotation != null) {
      name = annotation.value();

      if (name == null || name.length() == 0) {
        name = annotation.name();
      }
    }


    if (name == null || name.length() == 0) {
      name = autodetectProcessApplicationName();
    }

    return name;
  }

  /**
   * Override this method to autodetect an application name in case the
   * {@link ProcessApplication} annotation was used but without parameter.
   */
  protected abstract String autodetectProcessApplicationName();

  public  T execute(Callable callable) throws ProcessApplicationExecutionException {
    ClassLoader originalClassloader = ClassLoaderUtil.getContextClassloader();
    ClassLoader processApplicationClassloader = getProcessApplicationClassloader();

    try {
      ClassLoaderUtil.setContextClassloader(processApplicationClassloader);

      return callable.call();

    } catch (Exception e) {
      throw LOG.processApplicationExecutionException(e);
    } finally {
      ClassLoaderUtil.setContextClassloader(originalClassloader);
    }
  }

  public  T execute(Callable callable, InvocationContext invocationContext) throws ProcessApplicationExecutionException {
    // allows to hook into the invocation
    return execute(callable);
  }

  public ClassLoader getProcessApplicationClassloader() {
    // the default implementation uses the classloader that loaded
    // the application-provided subclass of this class.
    return ClassLoaderUtil.getClassloader(getClass());
  }

  public ProcessApplicationInterface getRawObject() {
    return this;
  }

  public Map getProperties() {
    return Collections.emptyMap();
  }

  public ELResolver getElResolver() {
    if (processApplicationElResolver == null) {
      synchronized (this) {
        if (processApplicationElResolver == null) {
          processApplicationElResolver = initProcessApplicationElResolver();
        }
      }
    }
    return processApplicationElResolver;

  }

  public BeanELResolver getBeanElResolver() {
    if (processApplicationBeanElResolver == null) {
      synchronized (this) {
        if (processApplicationBeanElResolver == null) {
          processApplicationBeanElResolver = new BeanELResolver();
        }
      }
    }
    return processApplicationBeanElResolver;
  }

  /**
   * 

Initializes the process application provided ElResolver. This implementation uses the * Java SE {@link ServiceLoader} facilities for resolving implementations of {@link ProcessApplicationElResolver}.

*

*

If you want to provide a custom implementation in your application, place a file named * META-INF/org.camunda.bpm.application.ProcessApplicationElResolver inside your application * which contains the fully qualified classname of your implementation. Or simply override this method.

* * @return the process application ElResolver. */ protected ELResolver initProcessApplicationElResolver() { return DefaultElResolverLookup.lookupResolver(this); } public ExecutionListener getExecutionListener() { return null; } public TaskListener getTaskListener() { return null; } /** * see {@link ProcessApplicationScriptEnvironment#getScriptEngineForName(String, boolean)} */ public ScriptEngine getScriptEngineForName(String name, boolean cache) { return getProcessApplicationScriptEnvironment().getScriptEngineForName(name, cache); } /** * see {@link ProcessApplicationScriptEnvironment#getEnvironmentScripts()} */ public Map> getEnvironmentScripts() { return getProcessApplicationScriptEnvironment().getEnvironmentScripts(); } protected ProcessApplicationScriptEnvironment getProcessApplicationScriptEnvironment() { if (processApplicationScriptEnvironment == null) { synchronized (this) { if (processApplicationScriptEnvironment == null) { processApplicationScriptEnvironment = new ProcessApplicationScriptEnvironment(this); } } } return processApplicationScriptEnvironment; } public VariableSerializers getVariableSerializers() { return variableSerializers; } public void setVariableSerializers(VariableSerializers variableSerializers) { this.variableSerializers = variableSerializers; } /** *

Provides the default Process Engine name to deploy to, if no Process Engine * was defined in processes.xml.

* * @return the default deploy-to Process Engine name. * The default value is "default". */ public String getDefaultDeployToEngineName() { return defaultDeployToEngineName; } /** *

Programmatically set the name of the Process Engine to deploy to if no Process Engine * is defined in processes.xml. This allows to circumvent the "default" Process * Engine name and set a custom one.

* * @param defaultDeployToEngineName */ protected void setDefaultDeployToEngineName(String defaultDeployToEngineName) { this.defaultDeployToEngineName = defaultDeployToEngineName; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy