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

org.camunda.bpm.engine.impl.persistence.entity.TimerEntity Maven / Gradle / Ivy

There is a newer version: 7.22.0-alpha5
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.engine.impl.persistence.entity;

import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.HashMap;
import java.util.Map;

import org.camunda.bpm.engine.impl.ProcessEngineLogger;
import org.camunda.bpm.engine.impl.calendar.BusinessCalendar;
import org.camunda.bpm.engine.impl.calendar.CycleBusinessCalendar;
import org.camunda.bpm.engine.impl.cfg.ProcessEngineConfigurationImpl;
import org.camunda.bpm.engine.impl.cfg.TransactionState;
import org.camunda.bpm.engine.impl.context.Context;
import org.camunda.bpm.engine.impl.interceptor.CommandContext;
import org.camunda.bpm.engine.impl.interceptor.CommandExecutor;
import org.camunda.bpm.engine.impl.jobexecutor.RepeatingFailedJobListener;
import org.camunda.bpm.engine.impl.jobexecutor.TimerDeclarationImpl;
import org.camunda.bpm.engine.impl.jobexecutor.TimerDeclarationType;
import org.camunda.bpm.engine.impl.jobexecutor.TimerEventJobHandler;
import org.camunda.bpm.engine.impl.jobexecutor.TimerEventJobHandler.TimerJobConfiguration;
import org.camunda.bpm.engine.impl.util.ClockUtil;


/**
 * @author Tom Baeyens
 */
public class TimerEntity extends JobEntity {

  protected static final String CYCLE_EXPRESSION_START_TYPE_1 = TimerDeclarationType.CYCLE + ": #";
  protected static final String CYCLE_EXPRESSION_START_TYPE_2 = TimerDeclarationType.CYCLE + ": $";
  public static final SimpleDateFormat SIMPLE_DATE_FORMAT = new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ss");

  public static final String TYPE = "timer";

  private static final long serialVersionUID = 1L;

  protected String repeat;

  protected long repeatOffset;

  public TimerEntity() {
  }

  public TimerEntity(TimerDeclarationImpl timerDeclaration) {
    repeat = timerDeclaration.getRepeat();
  }

  protected TimerEntity(TimerEntity te) {
    jobHandlerConfiguration = te.jobHandlerConfiguration;
    jobHandlerType = te.jobHandlerType;
    isExclusive = te.isExclusive;
    repeat = te.repeat;
    repeatOffset = te.repeatOffset;
    retries = te.retries;
    executionId = te.executionId;
    processInstanceId = te.processInstanceId;
    jobDefinitionId = te.jobDefinitionId;
    suspensionState = te.suspensionState;
    deploymentId = te.deploymentId;
    processDefinitionId = te.processDefinitionId;
    processDefinitionKey = te.processDefinitionKey;
    tenantId = te.tenantId;
    priority = te.priority;
  }

  @Override
  protected void preExecute(CommandContext commandContext) {
    if (getJobHandler() instanceof TimerEventJobHandler) {
      TimerJobConfiguration configuration = (TimerJobConfiguration) getJobHandlerConfiguration();
      if (repeat != null && !configuration.isFollowUpJobCreated()) {
        // this timer is a repeating timer and
        // a follow up timer job has not been scheduled yet

        // when reevaluateTimeCycleWhenDue is enabled and cycle is an expression
        if (isReevaluateTimeCycleWhenDue(commandContext) && isCycleExpression()) {
          String expressionValue = parseExpression(commandContext);
          if (expressionValue != null) {
            repeat = adjustRepeatBasedOnNewExpression(expressionValue);
          }
        }
        Date newDueDate = calculateNewDueDate();

        if (newDueDate != null) {
          // the listener is added to the transaction as SYNC on ROLLABCK,
          // when it is necessary to schedule a new timer job invocation.
          // If the transaction does not rollback, it is ignored.
          ProcessEngineConfigurationImpl processEngineConfiguration = Context.getProcessEngineConfiguration();
          CommandExecutor commandExecutor = processEngineConfiguration.getCommandExecutorTxRequiresNew();
          RepeatingFailedJobListener listener = createRepeatingFailedJobListener(commandExecutor);

          commandContext.getTransactionContext().addTransactionListener(
              TransactionState.ROLLED_BACK,
              listener);

          // create a new timer job
          createNewTimerJob(newDueDate);
        }
      }
    }
  }

  protected String parseExpression(CommandContext commandContext) {
    String expressionValue = null;
    String expression = jobDefinition.getJobConfiguration().substring(CYCLE_EXPRESSION_START_TYPE_1.length() - 1);
    try {
      expressionValue = commandContext.getProcessEngineConfiguration()
          .getExpressionManager()
          .createExpression(expression)
          .getValue(execution).toString();
    } catch (Exception e) {
      throw ProcessEngineLogger.UTIL_LOGGER.exceptionWhileParsingCycleExpresison(expression, e);
    }
    return expressionValue;
  }

  protected boolean isReevaluateTimeCycleWhenDue(CommandContext commandContext) {
    return commandContext.getProcessEngineConfiguration().isReevaluateTimeCycleWhenDue();
  }

  protected boolean isCycleExpression() {
    // Note timer cycle configuration is constructed in BpmnParse#parseTimer
    String jobConfiguration = jobDefinition.getJobConfiguration();
    return jobConfiguration.contains(CYCLE_EXPRESSION_START_TYPE_1)
        || jobConfiguration.contains(CYCLE_EXPRESSION_START_TYPE_2);
  }

  protected String adjustRepeatBasedOnNewExpression(String expressionValue) {
    String changedRepeat;
    if (expressionValue.startsWith("R")) { // changed to a repeatable interval
      if (repeat.startsWith("R") ) {
        if (isSameRepeatCycle(expressionValue)) {
          // the same repeatable interval => keep the start date
          changedRepeat = repeat;
        } else {
          // different repeatable interval => change the start date
          changedRepeat = replaceRepeatCycleAndDate(expressionValue);
        }
      } else {
        // was a cron expression => change the start date
        changedRepeat = replaceRepeatCycleAndDate(expressionValue);
      }
    } else {
      // changed to a cron expression
      changedRepeat = expressionValue;
    }
    return changedRepeat;
  }

  protected boolean isSameRepeatCycle(String expressionValue) {
    String[] currentRepeat = repeat.split("/");      // "R3/date/PT2H"
    String[] newRepeat = expressionValue.split("/"); // "R3/PT2H" or "R3/date/PT2H"
    if (currentRepeat.length == 3 && newRepeat.length == 2) {
      return currentRepeat[0].equals(newRepeat[0]) && currentRepeat[2].equals(newRepeat[1]);
    } else if (currentRepeat.length == 3 && newRepeat.length == 3) {
      return repeat.equals(expressionValue);
    } else {
      // incorrect cycle => keep the existing one
      return true;
    }
  }

  public static String replaceRepeatCycleAndDate(String repeatExpression) {
    if (repeatExpression.split("/").length == 2) {
      return repeatExpression.replace("/", "/" + SIMPLE_DATE_FORMAT.format(ClockUtil.getCurrentTime()) + "/");
    }
    return repeatExpression; // expression include start date
  }

  protected RepeatingFailedJobListener createRepeatingFailedJobListener(CommandExecutor commandExecutor) {
    return new RepeatingFailedJobListener(commandExecutor, getId());
  }

  public void createNewTimerJob(Date dueDate) {
    // create new timer job
    TimerEntity newTimer = new TimerEntity(this);
    newTimer.setDuedate(dueDate);
    Context
      .getCommandContext()
      .getJobManager()
      .schedule(newTimer);
  }

  public Date calculateNewDueDate() {
    BusinessCalendar businessCalendar = Context
        .getProcessEngineConfiguration()
        .getBusinessCalendarManager()
        .getBusinessCalendar(CycleBusinessCalendar.NAME);
    return ((CycleBusinessCalendar) businessCalendar).resolveDuedate(repeat, null, repeatOffset);
  }

  public String getRepeat() {
    return repeat;
  }

  public void setRepeat(String repeat) {
    this.repeat = repeat;
  }

  public long getRepeatOffset() {
    return repeatOffset;
  }

  public void setRepeatOffset(long repeatOffset) {
    this.repeatOffset = repeatOffset;
  }

  @Override
  public String getType() {
    return TYPE;
  }

  @Override
  public Object getPersistentState() {
    Map persistentState = (HashMap) super.getPersistentState();
    persistentState.put("repeat", repeat);

    return persistentState;
  }

  @Override
  public String toString() {
    return this.getClass().getSimpleName()
           + "[repeat=" + repeat
           + ", id=" + id
           + ", revision=" + revision
           + ", duedate=" + duedate
           + ", repeatOffset=" + repeatOffset
           + ", lockOwner=" + lockOwner
           + ", lockExpirationTime=" + lockExpirationTime
           + ", executionId=" + executionId
           + ", processInstanceId=" + processInstanceId
           + ", isExclusive=" + isExclusive
           + ", retries=" + retries
           + ", jobHandlerType=" + jobHandlerType
           + ", jobHandlerConfiguration=" + jobHandlerConfiguration
           + ", exceptionByteArray=" + exceptionByteArray
           + ", exceptionByteArrayId=" + exceptionByteArrayId
           + ", exceptionMessage=" + exceptionMessage
           + ", deploymentId=" + deploymentId
           + "]";
  }

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy