Add gated execution mode for Batch2 (#3545)
* Add gated execution mode for Batcvh2 * Allow for async processing * Test fixes * Cleanup * Add javadoc * Fixes
This commit is contained in:
parent
51f38eae98
commit
5dfdc91682
|
@ -35,6 +35,7 @@ import org.springframework.data.domain.PageRequest;
|
||||||
import javax.annotation.Nonnull;
|
import javax.annotation.Nonnull;
|
||||||
import javax.transaction.Transactional;
|
import javax.transaction.Transactional;
|
||||||
import java.util.Date;
|
import java.util.Date;
|
||||||
|
import java.util.EnumSet;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Optional;
|
import java.util.Optional;
|
||||||
import java.util.UUID;
|
import java.util.UUID;
|
||||||
|
@ -91,6 +92,7 @@ public class JpaJobPersistenceImpl implements IJobPersistence {
|
||||||
entity.setDefinitionVersion(theInstance.getJobDefinitionVersion());
|
entity.setDefinitionVersion(theInstance.getJobDefinitionVersion());
|
||||||
entity.setStatus(theInstance.getStatus());
|
entity.setStatus(theInstance.getStatus());
|
||||||
entity.setParams(theInstance.getParameters());
|
entity.setParams(theInstance.getParameters());
|
||||||
|
entity.setCurrentGatedStepId(theInstance.getCurrentGatedStepId());
|
||||||
entity.setCreateTime(new Date());
|
entity.setCreateTime(new Date());
|
||||||
|
|
||||||
entity = myJobInstanceRepository.save(entity);
|
entity = myJobInstanceRepository.save(entity);
|
||||||
|
@ -156,6 +158,7 @@ public class JpaJobPersistenceImpl implements IJobPersistence {
|
||||||
retVal.setErrorCount(theEntity.getErrorCount());
|
retVal.setErrorCount(theEntity.getErrorCount());
|
||||||
retVal.setEstimatedTimeRemaining(theEntity.getEstimatedTimeRemaining());
|
retVal.setEstimatedTimeRemaining(theEntity.getEstimatedTimeRemaining());
|
||||||
retVal.setParameters(theEntity.getParams());
|
retVal.setParameters(theEntity.getParams());
|
||||||
|
retVal.setCurrentGatedStepId(theEntity.getCurrentGatedStepId());
|
||||||
return retVal;
|
return retVal;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -202,6 +205,7 @@ public class JpaJobPersistenceImpl implements IJobPersistence {
|
||||||
instance.setErrorMessage(theInstance.getErrorMessage());
|
instance.setErrorMessage(theInstance.getErrorMessage());
|
||||||
instance.setErrorCount(theInstance.getErrorCount());
|
instance.setErrorCount(theInstance.getErrorCount());
|
||||||
instance.setEstimatedTimeRemaining(theInstance.getEstimatedTimeRemaining());
|
instance.setEstimatedTimeRemaining(theInstance.getEstimatedTimeRemaining());
|
||||||
|
instance.setCurrentGatedStepId(theInstance.getCurrentGatedStepId());
|
||||||
|
|
||||||
myJobInstanceRepository.save(instance);
|
myJobInstanceRepository.save(instance);
|
||||||
}
|
}
|
||||||
|
|
|
@ -37,6 +37,7 @@ import org.springframework.beans.factory.annotation.Autowired;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
import java.util.function.Consumer;
|
||||||
import java.util.stream.Collectors;
|
import java.util.stream.Collectors;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -85,7 +86,8 @@ public class DeleteExpungeProcessor implements ItemProcessor<List<Long>, List<St
|
||||||
|
|
||||||
List<ResourceLink> conflictResourceLinks = Collections.synchronizedList(new ArrayList<>());
|
List<ResourceLink> conflictResourceLinks = Collections.synchronizedList(new ArrayList<>());
|
||||||
PartitionRunner partitionRunner = new PartitionRunner(PROCESS_NAME, THREAD_PREFIX, myDaoConfig.getExpungeBatchSize(), myDaoConfig.getExpungeThreadCount());
|
PartitionRunner partitionRunner = new PartitionRunner(PROCESS_NAME, THREAD_PREFIX, myDaoConfig.getExpungeBatchSize(), myDaoConfig.getExpungeThreadCount());
|
||||||
partitionRunner.runInPartitionedThreads(thePids, someTargetPids -> findResourceLinksWithTargetPidIn(thePids, someTargetPids, conflictResourceLinks));
|
Consumer<List<Long>> listConsumer = someTargetPids -> findResourceLinksWithTargetPidIn(thePids, someTargetPids, conflictResourceLinks);
|
||||||
|
partitionRunner.runInPartitionedThreads(thePids, listConsumer);
|
||||||
|
|
||||||
if (conflictResourceLinks.isEmpty()) {
|
if (conflictResourceLinks.isEmpty()) {
|
||||||
return;
|
return;
|
||||||
|
|
|
@ -38,6 +38,7 @@ import javax.persistence.TemporalType;
|
||||||
import java.io.Serializable;
|
import java.io.Serializable;
|
||||||
import java.util.Date;
|
import java.util.Date;
|
||||||
|
|
||||||
|
import static ca.uhn.fhir.batch2.model.JobDefinition.ID_MAX_LENGTH;
|
||||||
import static ca.uhn.fhir.jpa.entity.Batch2WorkChunkEntity.ERROR_MSG_MAX_LENGTH;
|
import static ca.uhn.fhir.jpa.entity.Batch2WorkChunkEntity.ERROR_MSG_MAX_LENGTH;
|
||||||
import static org.apache.commons.lang3.StringUtils.left;
|
import static org.apache.commons.lang3.StringUtils.left;
|
||||||
|
|
||||||
|
@ -101,6 +102,16 @@ public class Batch2JobInstanceEntity implements Serializable {
|
||||||
private int myErrorCount;
|
private int myErrorCount;
|
||||||
@Column(name = "EST_REMAINING", length = TIME_REMAINING_LENGTH, nullable = true)
|
@Column(name = "EST_REMAINING", length = TIME_REMAINING_LENGTH, nullable = true)
|
||||||
private String myEstimatedTimeRemaining;
|
private String myEstimatedTimeRemaining;
|
||||||
|
@Column(name = "CUR_GATED_STEP_ID", length = ID_MAX_LENGTH, nullable = true)
|
||||||
|
private String myCurrentGatedStepId;
|
||||||
|
|
||||||
|
public String getCurrentGatedStepId() {
|
||||||
|
return myCurrentGatedStepId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setCurrentGatedStepId(String theCurrentGatedStepId) {
|
||||||
|
myCurrentGatedStepId = theCurrentGatedStepId;
|
||||||
|
}
|
||||||
|
|
||||||
public boolean isCancelled() {
|
public boolean isCancelled() {
|
||||||
return myCancelled;
|
return myCancelled;
|
||||||
|
|
|
@ -49,11 +49,10 @@ import java.util.stream.Collectors;
|
||||||
@SuppressWarnings({"SqlNoDataSourceInspection", "SpellCheckingInspection"})
|
@SuppressWarnings({"SqlNoDataSourceInspection", "SpellCheckingInspection"})
|
||||||
public class HapiFhirJpaMigrationTasks extends BaseMigrationTasks<VersionEnum> {
|
public class HapiFhirJpaMigrationTasks extends BaseMigrationTasks<VersionEnum> {
|
||||||
|
|
||||||
private final Set<FlagEnum> myFlags;
|
|
||||||
|
|
||||||
// H2, Derby, MariaDB, and MySql automatically add indexes to foreign keys
|
// H2, Derby, MariaDB, and MySql automatically add indexes to foreign keys
|
||||||
public static final DriverTypeEnum[] NON_AUTOMATIC_FK_INDEX_PLATFORMS = new DriverTypeEnum[]{
|
public static final DriverTypeEnum[] NON_AUTOMATIC_FK_INDEX_PLATFORMS = new DriverTypeEnum[]{
|
||||||
DriverTypeEnum.POSTGRES_9_4, DriverTypeEnum.ORACLE_12C, DriverTypeEnum.MSSQL_2012};
|
DriverTypeEnum.POSTGRES_9_4, DriverTypeEnum.ORACLE_12C, DriverTypeEnum.MSSQL_2012};
|
||||||
|
private final Set<FlagEnum> myFlags;
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -274,14 +273,21 @@ public class HapiFhirJpaMigrationTasks extends BaseMigrationTasks<VersionEnum> {
|
||||||
replaceNumericSPIndices(version);
|
replaceNumericSPIndices(version);
|
||||||
replaceQuantitySPIndices(version);
|
replaceQuantitySPIndices(version);
|
||||||
|
|
||||||
// Drop Index on HFJ_RESOURCE.INDEX_STATUS
|
// Drop Index on HFJ_RESOURCE.INDEX_STATUS
|
||||||
version
|
version
|
||||||
.onTable("HFJ_RESOURCE")
|
.onTable("HFJ_RESOURCE")
|
||||||
.dropIndex("20220314.1", "IDX_INDEXSTATUS");
|
.dropIndex("20220314.1", "IDX_INDEXSTATUS");
|
||||||
|
|
||||||
|
version
|
||||||
|
.onTable("BT2_JOB_INSTANCE")
|
||||||
|
.addColumn("20220416.1", "CUR_GATED_STEP_ID")
|
||||||
|
.nullable()
|
||||||
|
.type(ColumnTypeEnum.STRING, 100);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* new numeric search indexing
|
* new numeric search indexing
|
||||||
|
*
|
||||||
* @see ca.uhn.fhir.jpa.search.builder.predicate.NumberPredicateBuilder
|
* @see ca.uhn.fhir.jpa.search.builder.predicate.NumberPredicateBuilder
|
||||||
* @see ca.uhn.fhir.jpa.model.entity.ResourceIndexedSearchParamNumber
|
* @see ca.uhn.fhir.jpa.model.entity.ResourceIndexedSearchParamNumber
|
||||||
*/
|
*/
|
||||||
|
@ -320,6 +326,7 @@ public class HapiFhirJpaMigrationTasks extends BaseMigrationTasks<VersionEnum> {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* new quantity search indexing
|
* new quantity search indexing
|
||||||
|
*
|
||||||
* @see ca.uhn.fhir.jpa.search.builder.predicate.QuantityPredicateBuilder
|
* @see ca.uhn.fhir.jpa.search.builder.predicate.QuantityPredicateBuilder
|
||||||
* @see ca.uhn.fhir.jpa.model.entity.ResourceIndexedSearchParamQuantity
|
* @see ca.uhn.fhir.jpa.model.entity.ResourceIndexedSearchParamQuantity
|
||||||
* @see ca.uhn.fhir.jpa.model.entity.ResourceIndexedSearchParamQuantityNormalized
|
* @see ca.uhn.fhir.jpa.model.entity.ResourceIndexedSearchParamQuantityNormalized
|
||||||
|
|
|
@ -20,7 +20,7 @@ package ca.uhn.fhir.jpa.test;
|
||||||
* #L%
|
* #L%
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import ca.uhn.fhir.batch2.api.IJobCleanerService;
|
import ca.uhn.fhir.batch2.api.IJobMaintenanceService;
|
||||||
import ca.uhn.fhir.batch2.api.IJobCoordinator;
|
import ca.uhn.fhir.batch2.api.IJobCoordinator;
|
||||||
import ca.uhn.fhir.batch2.model.JobInstance;
|
import ca.uhn.fhir.batch2.model.JobInstance;
|
||||||
import ca.uhn.fhir.batch2.model.StatusEnum;
|
import ca.uhn.fhir.batch2.model.StatusEnum;
|
||||||
|
@ -33,21 +33,21 @@ import static org.hamcrest.Matchers.equalTo;
|
||||||
public class Batch2JobHelper {
|
public class Batch2JobHelper {
|
||||||
|
|
||||||
@Autowired
|
@Autowired
|
||||||
private IJobCleanerService myJobCleanerService;
|
private IJobMaintenanceService myJobCleanerService;
|
||||||
|
|
||||||
@Autowired
|
@Autowired
|
||||||
private IJobCoordinator myJobCoordinator;
|
private IJobCoordinator myJobCoordinator;
|
||||||
|
|
||||||
public void awaitJobCompletion(String theId) {
|
public void awaitJobCompletion(String theId) {
|
||||||
await().until(() -> {
|
await().until(() -> {
|
||||||
myJobCleanerService.runCleanupPass();
|
myJobCleanerService.runMaintenancePass();
|
||||||
return myJobCoordinator.getInstance(theId).getStatus();
|
return myJobCoordinator.getInstance(theId).getStatus();
|
||||||
}, equalTo(StatusEnum.COMPLETED));
|
}, equalTo(StatusEnum.COMPLETED));
|
||||||
}
|
}
|
||||||
|
|
||||||
public JobInstance awaitJobFailure(String theId) {
|
public JobInstance awaitJobFailure(String theId) {
|
||||||
await().until(() -> {
|
await().until(() -> {
|
||||||
myJobCleanerService.runCleanupPass();
|
myJobCleanerService.runMaintenancePass();
|
||||||
return myJobCoordinator.getInstance(theId).getStatus();
|
return myJobCoordinator.getInstance(theId).getStatus();
|
||||||
}, Matchers.anyOf(equalTo(StatusEnum.ERRORED),equalTo(StatusEnum.FAILED)));
|
}, Matchers.anyOf(equalTo(StatusEnum.ERRORED),equalTo(StatusEnum.FAILED)));
|
||||||
return myJobCoordinator.getInstance(theId);
|
return myJobCoordinator.getInstance(theId);
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
package ca.uhn.fhir.jpa.bulk.imprt2;
|
package ca.uhn.fhir.jpa.bulk.imprt2;
|
||||||
|
|
||||||
import ca.uhn.fhir.batch2.api.IJobCleanerService;
|
import ca.uhn.fhir.batch2.api.IJobMaintenanceService;
|
||||||
import ca.uhn.fhir.batch2.api.IJobCoordinator;
|
import ca.uhn.fhir.batch2.api.IJobCoordinator;
|
||||||
import ca.uhn.fhir.batch2.jobs.imprt.BulkImportAppCtx;
|
import ca.uhn.fhir.batch2.jobs.imprt.BulkImportAppCtx;
|
||||||
import ca.uhn.fhir.batch2.jobs.imprt.BulkImportFileServlet;
|
import ca.uhn.fhir.batch2.jobs.imprt.BulkImportFileServlet;
|
||||||
|
@ -29,7 +29,6 @@ import org.slf4j.LoggerFactory;
|
||||||
import org.springframework.beans.factory.annotation.Autowired;
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
import org.springframework.beans.factory.annotation.Qualifier;
|
import org.springframework.beans.factory.annotation.Qualifier;
|
||||||
|
|
||||||
import java.io.StringReader;
|
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.stream.Collectors;
|
import java.util.stream.Collectors;
|
||||||
|
@ -56,7 +55,7 @@ public class BulkImportR4Test extends BaseJpaR4Test {
|
||||||
@Autowired
|
@Autowired
|
||||||
private IJobCoordinator myJobCoordinator;
|
private IJobCoordinator myJobCoordinator;
|
||||||
@Autowired
|
@Autowired
|
||||||
private IJobCleanerService myJobCleanerService;
|
private IJobMaintenanceService myJobCleanerService;
|
||||||
@Autowired
|
@Autowired
|
||||||
private IBatch2JobInstanceRepository myJobInstanceRepository;
|
private IBatch2JobInstanceRepository myJobInstanceRepository;
|
||||||
@Autowired
|
@Autowired
|
||||||
|
@ -99,7 +98,7 @@ public class BulkImportR4Test extends BaseJpaR4Test {
|
||||||
// Verify
|
// Verify
|
||||||
|
|
||||||
await().until(() -> {
|
await().until(() -> {
|
||||||
myJobCleanerService.runCleanupPass();
|
myJobCleanerService.runMaintenancePass();
|
||||||
JobInstance instance = myJobCoordinator.getInstance(instanceId);
|
JobInstance instance = myJobCoordinator.getInstance(instanceId);
|
||||||
return instance.getStatus();
|
return instance.getStatus();
|
||||||
}, equalTo(StatusEnum.COMPLETED));
|
}, equalTo(StatusEnum.COMPLETED));
|
||||||
|
@ -152,7 +151,7 @@ public class BulkImportR4Test extends BaseJpaR4Test {
|
||||||
// Verify
|
// Verify
|
||||||
|
|
||||||
await().until(() -> {
|
await().until(() -> {
|
||||||
myJobCleanerService.runCleanupPass();
|
myJobCleanerService.runMaintenancePass();
|
||||||
JobInstance instance = myJobCoordinator.getInstance(instanceId);
|
JobInstance instance = myJobCoordinator.getInstance(instanceId);
|
||||||
return instance.getStatus();
|
return instance.getStatus();
|
||||||
}, equalTo(StatusEnum.ERRORED));
|
}, equalTo(StatusEnum.ERRORED));
|
||||||
|
@ -174,7 +173,7 @@ public class BulkImportR4Test extends BaseJpaR4Test {
|
||||||
});
|
});
|
||||||
|
|
||||||
await().until(() -> {
|
await().until(() -> {
|
||||||
myJobCleanerService.runCleanupPass();
|
myJobCleanerService.runMaintenancePass();
|
||||||
JobInstance instance = myJobCoordinator.getInstance(instanceId);
|
JobInstance instance = myJobCoordinator.getInstance(instanceId);
|
||||||
return instance.getErrorCount();
|
return instance.getErrorCount();
|
||||||
}, equalTo(3));
|
}, equalTo(3));
|
||||||
|
@ -224,7 +223,7 @@ public class BulkImportR4Test extends BaseJpaR4Test {
|
||||||
// Verify
|
// Verify
|
||||||
|
|
||||||
await().until(() -> {
|
await().until(() -> {
|
||||||
myJobCleanerService.runCleanupPass();
|
myJobCleanerService.runMaintenancePass();
|
||||||
JobInstance instance = myJobCoordinator.getInstance(instanceId);
|
JobInstance instance = myJobCoordinator.getInstance(instanceId);
|
||||||
return instance.getStatus();
|
return instance.getStatus();
|
||||||
}, equalTo(StatusEnum.FAILED));
|
}, equalTo(StatusEnum.FAILED));
|
||||||
|
@ -267,7 +266,7 @@ public class BulkImportR4Test extends BaseJpaR4Test {
|
||||||
// Verify
|
// Verify
|
||||||
|
|
||||||
await().until(() -> {
|
await().until(() -> {
|
||||||
myJobCleanerService.runCleanupPass();
|
myJobCleanerService.runMaintenancePass();
|
||||||
JobInstance instance = myJobCoordinator.getInstance(instanceId);
|
JobInstance instance = myJobCoordinator.getInstance(instanceId);
|
||||||
return instance.getStatus();
|
return instance.getStatus();
|
||||||
}, equalTo(StatusEnum.FAILED));
|
}, equalTo(StatusEnum.FAILED));
|
||||||
|
|
|
@ -24,6 +24,7 @@ import static org.hamcrest.Matchers.hasSize;
|
||||||
import static org.junit.jupiter.api.Assertions.assertEquals;
|
import static org.junit.jupiter.api.Assertions.assertEquals;
|
||||||
|
|
||||||
public class ReindexJobTest extends BaseJpaR4Test {
|
public class ReindexJobTest extends BaseJpaR4Test {
|
||||||
|
|
||||||
@Autowired
|
@Autowired
|
||||||
private IJobCoordinator myJobCoordinator;
|
private IJobCoordinator myJobCoordinator;
|
||||||
|
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
<configuration>
|
<configuration>
|
||||||
<appender name="STDOUT" class="ch.qos.logback.core.ConsoleAppender">
|
<appender name="STDOUT" class="ch.qos.logback.core.ConsoleAppender">
|
||||||
<filter class="ch.qos.logback.classic.filter.ThresholdFilter">
|
<filter class="ch.qos.logback.classic.filter.ThresholdFilter">
|
||||||
<level>TRACE</level>
|
<level>INFO</level>
|
||||||
</filter>
|
</filter>
|
||||||
<encoder>
|
<encoder>
|
||||||
<pattern>%d{yyyy-MM-dd HH:mm:ss.SSS} [%thread] %-5level %logger{36} [%file:%line] %msg%n</pattern>
|
<pattern>%d{yyyy-MM-dd HH:mm:ss.SSS} [%thread] %-5level %logger{36} [%file:%line] %msg%n</pattern>
|
||||||
|
@ -23,14 +23,6 @@
|
||||||
<logger name="org.eclipse.jetty.websocket" additivity="false" level="info">
|
<logger name="org.eclipse.jetty.websocket" additivity="false" level="info">
|
||||||
<appender-ref ref="STDOUT" />
|
<appender-ref ref="STDOUT" />
|
||||||
</logger>
|
</logger>
|
||||||
<!-- TODO GGG HS can probably eventually disable this. -->
|
|
||||||
<logger name="org.hibernate.search.backend.lucene.infostream" additivity="true" level="trace">
|
|
||||||
<appender-ref ref="STDOUT" />
|
|
||||||
</logger>
|
|
||||||
<logger name="org.hibernate.search.query" additivity="true" level="trace">
|
|
||||||
<appender-ref ref="STDOUT" />
|
|
||||||
</logger>
|
|
||||||
j
|
|
||||||
<logger name="org.hibernate.event.internal.DefaultPersistEventListener" additivity="true" level="info">
|
<logger name="org.hibernate.event.internal.DefaultPersistEventListener" additivity="true" level="info">
|
||||||
<appender-ref ref="STDOUT" />
|
<appender-ref ref="STDOUT" />
|
||||||
</logger>
|
</logger>
|
||||||
|
|
|
@ -41,6 +41,7 @@ public class ReindexAppCtx {
|
||||||
.setJobDefinitionVersion(1)
|
.setJobDefinitionVersion(1)
|
||||||
.setParametersType(ReindexJobParameters.class)
|
.setParametersType(ReindexJobParameters.class)
|
||||||
.setParametersValidator(reindexJobParametersValidator())
|
.setParametersValidator(reindexJobParametersValidator())
|
||||||
|
.gatedExecution()
|
||||||
.addFirstStep(
|
.addFirstStep(
|
||||||
"generate-ranges",
|
"generate-ranges",
|
||||||
"Generate data ranges to reindex",
|
"Generate data ranges to reindex",
|
||||||
|
|
|
@ -0,0 +1,9 @@
|
||||||
|
package ca.uhn.fhir.batch2.api;
|
||||||
|
|
||||||
|
import ca.uhn.fhir.model.api.IModelJson;
|
||||||
|
|
||||||
|
public interface IJobCompletionHandler<PT extends IModelJson> {
|
||||||
|
|
||||||
|
void jobComplete(JobCompletionDetails<PT> theDetails);
|
||||||
|
|
||||||
|
}
|
|
@ -54,5 +54,4 @@ public interface IJobCoordinator {
|
||||||
|
|
||||||
void cancelInstance(String theInstanceId) throws ResourceNotFoundException;
|
void cancelInstance(String theInstanceId) throws ResourceNotFoundException;
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -20,8 +20,8 @@ package ca.uhn.fhir.batch2.api;
|
||||||
* #L%
|
* #L%
|
||||||
*/
|
*/
|
||||||
|
|
||||||
public interface IJobCleanerService {
|
public interface IJobMaintenanceService {
|
||||||
|
|
||||||
void runCleanupPass();
|
void runMaintenancePass();
|
||||||
|
|
||||||
}
|
}
|
|
@ -101,7 +101,7 @@ public interface IJobPersistence {
|
||||||
/**
|
/**
|
||||||
* Increments the work chunk error count by the given amount
|
* Increments the work chunk error count by the given amount
|
||||||
*
|
*
|
||||||
* @param theChunkId The chunk ID
|
* @param theChunkId The chunk ID
|
||||||
* @param theIncrementBy The number to increment the error count by
|
* @param theIncrementBy The number to increment the error count by
|
||||||
*/
|
*/
|
||||||
void incrementWorkChunkErrorCount(String theChunkId, int theIncrementBy);
|
void incrementWorkChunkErrorCount(String theChunkId, int theIncrementBy);
|
||||||
|
|
|
@ -0,0 +1,56 @@
|
||||||
|
package ca.uhn.fhir.batch2.api;
|
||||||
|
|
||||||
|
/*-
|
||||||
|
* #%L
|
||||||
|
* HAPI FHIR JPA Server - Batch2 Task Processor
|
||||||
|
* %%
|
||||||
|
* Copyright (C) 2014 - 2022 Smile CDR, Inc.
|
||||||
|
* %%
|
||||||
|
* 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.
|
||||||
|
* #L%
|
||||||
|
*/
|
||||||
|
|
||||||
|
import ca.uhn.fhir.model.api.IModelJson;
|
||||||
|
import org.apache.commons.lang3.Validate;
|
||||||
|
|
||||||
|
import javax.annotation.Nonnull;
|
||||||
|
|
||||||
|
public class JobCompletionDetails<PT extends IModelJson> {
|
||||||
|
|
||||||
|
private final PT myParameters;
|
||||||
|
private final String myInstanceId;
|
||||||
|
|
||||||
|
public JobCompletionDetails(@Nonnull PT theParameters, @Nonnull String theInstanceId) {
|
||||||
|
Validate.notNull(theParameters);
|
||||||
|
myParameters = theParameters;
|
||||||
|
myInstanceId = theInstanceId;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the parameters associated with this job instance. Note that parameters
|
||||||
|
* are set when the job instance is created and can not be modified after that.
|
||||||
|
*/
|
||||||
|
@Nonnull
|
||||||
|
public PT getParameters() {
|
||||||
|
return myParameters;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the job instance ID being executed
|
||||||
|
*/
|
||||||
|
@Nonnull
|
||||||
|
public String getInstanceId() {
|
||||||
|
return myInstanceId;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -6,7 +6,7 @@
|
||||||
* framework, used to start and stop jobs and inquire about status.
|
* framework, used to start and stop jobs and inquire about status.
|
||||||
* </ol>
|
* </ol>
|
||||||
* <ol>
|
* <ol>
|
||||||
* {@link ca.uhn.fhir.batch2.api.IJobCleanerService} is a background processor that
|
* {@link ca.uhn.fhir.batch2.api.IJobMaintenanceService} is a background processor that
|
||||||
* updates statistics and clean up stale data
|
* updates statistics and clean up stale data
|
||||||
* </ol>
|
* </ol>
|
||||||
* <ol>
|
* <ol>
|
||||||
|
|
|
@ -20,11 +20,11 @@ package ca.uhn.fhir.batch2.config;
|
||||||
* #L%
|
* #L%
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import ca.uhn.fhir.batch2.api.IJobCleanerService;
|
import ca.uhn.fhir.batch2.api.IJobMaintenanceService;
|
||||||
import ca.uhn.fhir.batch2.api.IJobCoordinator;
|
import ca.uhn.fhir.batch2.api.IJobCoordinator;
|
||||||
import ca.uhn.fhir.batch2.api.IJobPersistence;
|
import ca.uhn.fhir.batch2.api.IJobPersistence;
|
||||||
import ca.uhn.fhir.batch2.impl.BatchJobSender;
|
import ca.uhn.fhir.batch2.impl.BatchJobSender;
|
||||||
import ca.uhn.fhir.batch2.impl.JobCleanerServiceImpl;
|
import ca.uhn.fhir.batch2.impl.JobMaintenanceServiceImpl;
|
||||||
import ca.uhn.fhir.batch2.impl.JobCoordinatorImpl;
|
import ca.uhn.fhir.batch2.impl.JobCoordinatorImpl;
|
||||||
import ca.uhn.fhir.batch2.impl.JobDefinitionRegistry;
|
import ca.uhn.fhir.batch2.impl.JobDefinitionRegistry;
|
||||||
import ca.uhn.fhir.batch2.model.JobWorkNotificationJsonMessage;
|
import ca.uhn.fhir.batch2.model.JobWorkNotificationJsonMessage;
|
||||||
|
@ -48,9 +48,14 @@ public abstract class BaseBatch2Config {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Bean
|
@Bean
|
||||||
public IJobCoordinator batch2JobCoordinator(IChannelFactory theChannelFactory, IJobPersistence theJobInstancePersister, JobDefinitionRegistry theJobDefinitionRegistry) {
|
public BatchJobSender batchJobSender(IChannelFactory theChannelFactory) {
|
||||||
|
return new BatchJobSender(batch2ProcessingChannelProducer(theChannelFactory));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
public IJobCoordinator batch2JobCoordinator(IChannelFactory theChannelFactory, IJobPersistence theJobInstancePersister, JobDefinitionRegistry theJobDefinitionRegistry, BatchJobSender theBatchJobSender) {
|
||||||
return new JobCoordinatorImpl(
|
return new JobCoordinatorImpl(
|
||||||
new BatchJobSender(batch2ProcessingChannelProducer(theChannelFactory)),
|
theBatchJobSender,
|
||||||
batch2ProcessingChannelReceiver(theChannelFactory),
|
batch2ProcessingChannelReceiver(theChannelFactory),
|
||||||
theJobInstancePersister,
|
theJobInstancePersister,
|
||||||
theJobDefinitionRegistry
|
theJobDefinitionRegistry
|
||||||
|
@ -58,8 +63,8 @@ public abstract class BaseBatch2Config {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Bean
|
@Bean
|
||||||
public IJobCleanerService batch2JobCleaner(ISchedulerService theSchedulerService, IJobPersistence theJobPersistence) {
|
public IJobMaintenanceService batch2JobMaintenanceService(ISchedulerService theSchedulerService, IJobPersistence theJobPersistence, JobDefinitionRegistry theJobDefinitionRegistry, BatchJobSender theBatchJobSender) {
|
||||||
return new JobCleanerServiceImpl(theSchedulerService, theJobPersistence);
|
return new JobMaintenanceServiceImpl(theSchedulerService, theJobPersistence, theJobDefinitionRegistry, theBatchJobSender);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Bean
|
@Bean
|
||||||
|
|
|
@ -37,6 +37,7 @@ import ca.uhn.fhir.batch2.model.JobWorkNotificationJsonMessage;
|
||||||
import ca.uhn.fhir.batch2.model.StatusEnum;
|
import ca.uhn.fhir.batch2.model.StatusEnum;
|
||||||
import ca.uhn.fhir.batch2.model.WorkChunk;
|
import ca.uhn.fhir.batch2.model.WorkChunk;
|
||||||
import ca.uhn.fhir.i18n.Msg;
|
import ca.uhn.fhir.i18n.Msg;
|
||||||
|
import ca.uhn.fhir.jpa.model.sched.ISchedulerService;
|
||||||
import ca.uhn.fhir.jpa.subscription.channel.api.IChannelReceiver;
|
import ca.uhn.fhir.jpa.subscription.channel.api.IChannelReceiver;
|
||||||
import ca.uhn.fhir.model.api.IModelJson;
|
import ca.uhn.fhir.model.api.IModelJson;
|
||||||
import ca.uhn.fhir.model.api.annotation.PasswordField;
|
import ca.uhn.fhir.model.api.annotation.PasswordField;
|
||||||
|
@ -49,6 +50,7 @@ import com.fasterxml.jackson.annotation.JsonProperty;
|
||||||
import org.apache.commons.lang3.Validate;
|
import org.apache.commons.lang3.Validate;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
import org.springframework.messaging.Message;
|
import org.springframework.messaging.Message;
|
||||||
import org.springframework.messaging.MessageHandler;
|
import org.springframework.messaging.MessageHandler;
|
||||||
import org.springframework.messaging.MessagingException;
|
import org.springframework.messaging.MessagingException;
|
||||||
|
@ -78,15 +80,13 @@ public class JobCoordinatorImpl extends BaseJobService implements IJobCoordinato
|
||||||
private final JobDefinitionRegistry myJobDefinitionRegistry;
|
private final JobDefinitionRegistry myJobDefinitionRegistry;
|
||||||
private final MessageHandler myReceiverHandler = new WorkChannelMessageHandler();
|
private final MessageHandler myReceiverHandler = new WorkChannelMessageHandler();
|
||||||
private final ValidatorFactory myValidatorFactory = Validation.buildDefaultValidatorFactory();
|
private final ValidatorFactory myValidatorFactory = Validation.buildDefaultValidatorFactory();
|
||||||
|
@Autowired
|
||||||
|
private ISchedulerService mySchedulerService;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Constructor
|
* Constructor
|
||||||
*/
|
*/
|
||||||
public JobCoordinatorImpl(
|
public JobCoordinatorImpl(@Nonnull BatchJobSender theBatchJobSender, @Nonnull IChannelReceiver theWorkChannelReceiver, @Nonnull IJobPersistence theJobPersistence, @Nonnull JobDefinitionRegistry theJobDefinitionRegistry) {
|
||||||
@Nonnull BatchJobSender theBatchJobSender,
|
|
||||||
@Nonnull IChannelReceiver theWorkChannelReceiver,
|
|
||||||
@Nonnull IJobPersistence theJobPersistence,
|
|
||||||
@Nonnull JobDefinitionRegistry theJobDefinitionRegistry) {
|
|
||||||
super(theJobPersistence);
|
super(theJobPersistence);
|
||||||
myBatchJobSender = theBatchJobSender;
|
myBatchJobSender = theBatchJobSender;
|
||||||
myWorkChannelReceiver = theWorkChannelReceiver;
|
myWorkChannelReceiver = theWorkChannelReceiver;
|
||||||
|
@ -95,9 +95,7 @@ public class JobCoordinatorImpl extends BaseJobService implements IJobCoordinato
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public String startInstance(JobInstanceStartRequest theStartRequest) {
|
public String startInstance(JobInstanceStartRequest theStartRequest) {
|
||||||
JobDefinition<?> jobDefinition = myJobDefinitionRegistry
|
JobDefinition<?> jobDefinition = myJobDefinitionRegistry.getLatestJobDefinition(theStartRequest.getJobDefinitionId()).orElseThrow(() -> new IllegalArgumentException(Msg.code(2063) + "Unknown job definition ID: " + theStartRequest.getJobDefinitionId()));
|
||||||
.getLatestJobDefinition(theStartRequest.getJobDefinitionId())
|
|
||||||
.orElseThrow(() -> new IllegalArgumentException(Msg.code(2063) + "Unknown job definition ID: " + theStartRequest.getJobDefinitionId()));
|
|
||||||
|
|
||||||
if (isBlank(theStartRequest.getParameters())) {
|
if (isBlank(theStartRequest.getParameters())) {
|
||||||
throw new InvalidRequestException(Msg.code(2065) + "No parameters supplied");
|
throw new InvalidRequestException(Msg.code(2065) + "No parameters supplied");
|
||||||
|
@ -115,6 +113,10 @@ public class JobCoordinatorImpl extends BaseJobService implements IJobCoordinato
|
||||||
instance.setStatus(StatusEnum.QUEUED);
|
instance.setStatus(StatusEnum.QUEUED);
|
||||||
instance.setParameters(theStartRequest.getParameters());
|
instance.setParameters(theStartRequest.getParameters());
|
||||||
|
|
||||||
|
if (jobDefinition.isGatedExecution()) {
|
||||||
|
instance.setCurrentGatedStepId(firstStepId);
|
||||||
|
}
|
||||||
|
|
||||||
String instanceId = myJobPersistence.storeNewInstance(instance);
|
String instanceId = myJobPersistence.storeNewInstance(instance);
|
||||||
|
|
||||||
BatchWorkChunk batchWorkChunk = new BatchWorkChunk(jobDefinitionId, jobDefinitionVersion, firstStepId, instanceId, 0, null);
|
BatchWorkChunk batchWorkChunk = new BatchWorkChunk(jobDefinitionId, jobDefinitionVersion, firstStepId, instanceId, 0, null);
|
||||||
|
@ -132,11 +134,7 @@ public class JobCoordinatorImpl extends BaseJobService implements IJobCoordinato
|
||||||
Validator validator = myValidatorFactory.getValidator();
|
Validator validator = myValidatorFactory.getValidator();
|
||||||
PT parameters = theStartRequest.getParameters(theJobDefinition.getParametersType());
|
PT parameters = theStartRequest.getParameters(theJobDefinition.getParametersType());
|
||||||
Set<ConstraintViolation<IModelJson>> constraintErrors = validator.validate(parameters);
|
Set<ConstraintViolation<IModelJson>> constraintErrors = validator.validate(parameters);
|
||||||
List<String> errorStrings = constraintErrors
|
List<String> errorStrings = constraintErrors.stream().map(t -> t.getPropertyPath() + " - " + t.getMessage()).sorted().collect(Collectors.toList());
|
||||||
.stream()
|
|
||||||
.map(t -> t.getPropertyPath() + " - " + t.getMessage())
|
|
||||||
.sorted()
|
|
||||||
.collect(Collectors.toList());
|
|
||||||
|
|
||||||
// Programmatic Validator
|
// Programmatic Validator
|
||||||
IJobParametersValidator<PT> parametersValidator = theJobDefinition.getParametersValidator();
|
IJobParametersValidator<PT> parametersValidator = theJobDefinition.getParametersValidator();
|
||||||
|
@ -147,13 +145,7 @@ public class JobCoordinatorImpl extends BaseJobService implements IJobCoordinato
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!errorStrings.isEmpty()) {
|
if (!errorStrings.isEmpty()) {
|
||||||
String message = "Failed to validate parameters for job of type " +
|
String message = "Failed to validate parameters for job of type " + theJobDefinition.getJobDefinitionId() + ": " + errorStrings.stream().map(t -> "\n * " + t).collect(Collectors.joining());
|
||||||
theJobDefinition.getJobDefinitionId() +
|
|
||||||
": " +
|
|
||||||
errorStrings
|
|
||||||
.stream()
|
|
||||||
.map(t -> "\n * " + t)
|
|
||||||
.collect(Collectors.joining());
|
|
||||||
|
|
||||||
throw new InvalidRequestException(Msg.code(2039) + message);
|
throw new InvalidRequestException(Msg.code(2039) + message);
|
||||||
}
|
}
|
||||||
|
@ -161,19 +153,12 @@ public class JobCoordinatorImpl extends BaseJobService implements IJobCoordinato
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public JobInstance getInstance(String theInstanceId) {
|
public JobInstance getInstance(String theInstanceId) {
|
||||||
return myJobPersistence
|
return myJobPersistence.fetchInstance(theInstanceId).map(t -> massageInstanceForUserAccess(t)).orElseThrow(() -> new ResourceNotFoundException(Msg.code(2040) + "Unknown instance ID: " + UrlUtil.escapeUrlParam(theInstanceId)));
|
||||||
.fetchInstance(theInstanceId)
|
|
||||||
.map(t -> massageInstanceForUserAccess(t))
|
|
||||||
.orElseThrow(() -> new ResourceNotFoundException(Msg.code(2040) + "Unknown instance ID: " + UrlUtil.escapeUrlParam(theInstanceId)));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public List<JobInstance> getInstances(int thePageSize, int thePageIndex) {
|
public List<JobInstance> getInstances(int thePageSize, int thePageIndex) {
|
||||||
return myJobPersistence
|
return myJobPersistence.fetchInstances(thePageSize, thePageIndex).stream().map(t -> massageInstanceForUserAccess(t)).collect(Collectors.toList());
|
||||||
.fetchInstances(thePageSize, thePageIndex)
|
|
||||||
.stream()
|
|
||||||
.map(t -> massageInstanceForUserAccess(t))
|
|
||||||
.collect(Collectors.toList());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -195,7 +180,7 @@ public class JobCoordinatorImpl extends BaseJobService implements IJobCoordinato
|
||||||
return retVal;
|
return retVal;
|
||||||
}
|
}
|
||||||
|
|
||||||
private <PT extends IModelJson, IT extends IModelJson, OT extends IModelJson> boolean executeStep(@Nonnull WorkChunk theWorkChunk, String jobDefinitionId, String targetStepId, Class<IT> theInputType, PT parameters, IJobStepWorker<PT, IT, OT> worker, BaseDataSink<OT> dataSink) {
|
private <PT extends IModelJson, IT extends IModelJson, OT extends IModelJson> boolean executeStep(@Nonnull WorkChunk theWorkChunk, String theJobDefinitionId, String theTargetStepId, Class<IT> theInputType, PT theParameters, IJobStepWorker<PT, IT, OT> theWorker, BaseDataSink<OT> theDataSink) {
|
||||||
IT data = null;
|
IT data = null;
|
||||||
if (!theInputType.equals(VoidModel.class)) {
|
if (!theInputType.equals(VoidModel.class)) {
|
||||||
data = theWorkChunk.getData(theInputType);
|
data = theWorkChunk.getData(theInputType);
|
||||||
|
@ -204,21 +189,21 @@ public class JobCoordinatorImpl extends BaseJobService implements IJobCoordinato
|
||||||
String instanceId = theWorkChunk.getInstanceId();
|
String instanceId = theWorkChunk.getInstanceId();
|
||||||
String chunkId = theWorkChunk.getId();
|
String chunkId = theWorkChunk.getId();
|
||||||
|
|
||||||
StepExecutionDetails<PT, IT> stepExecutionDetails = new StepExecutionDetails<>(parameters, data, instanceId, chunkId);
|
StepExecutionDetails<PT, IT> stepExecutionDetails = new StepExecutionDetails<>(theParameters, data, instanceId, chunkId);
|
||||||
RunOutcome outcome;
|
RunOutcome outcome;
|
||||||
try {
|
try {
|
||||||
outcome = worker.run(stepExecutionDetails, dataSink);
|
outcome = theWorker.run(stepExecutionDetails, theDataSink);
|
||||||
Validate.notNull(outcome, "Step worker returned null: %s", worker.getClass());
|
Validate.notNull(outcome, "Step theWorker returned null: %s", theWorker.getClass());
|
||||||
} catch (JobExecutionFailedException e) {
|
} catch (JobExecutionFailedException e) {
|
||||||
ourLog.error("Unrecoverable failure executing job {} step {}", jobDefinitionId, targetStepId, e);
|
ourLog.error("Unrecoverable failure executing job {} step {}", theJobDefinitionId, theTargetStepId, e);
|
||||||
myJobPersistence.markWorkChunkAsFailed(chunkId, e.toString());
|
myJobPersistence.markWorkChunkAsFailed(chunkId, e.toString());
|
||||||
return false;
|
return false;
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
ourLog.error("Failure executing job {} step {}", jobDefinitionId, targetStepId, e);
|
ourLog.error("Failure executing job {} step {}", theJobDefinitionId, theTargetStepId, e);
|
||||||
myJobPersistence.markWorkChunkAsErroredAndIncrementErrorCount(chunkId, e.toString());
|
myJobPersistence.markWorkChunkAsErroredAndIncrementErrorCount(chunkId, e.toString());
|
||||||
throw new JobExecutionFailedException(Msg.code(2041) + e.getMessage(), e);
|
throw new JobExecutionFailedException(Msg.code(2041) + e.getMessage(), e);
|
||||||
} catch (Throwable t) {
|
} catch (Throwable t) {
|
||||||
ourLog.error("Unexpected failure executing job {} step {}", jobDefinitionId, targetStepId, t);
|
ourLog.error("Unexpected failure executing job {} step {}", theJobDefinitionId, theTargetStepId, t);
|
||||||
myJobPersistence.markWorkChunkAsFailed(chunkId, t.toString());
|
myJobPersistence.markWorkChunkAsFailed(chunkId, t.toString());
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
@ -226,7 +211,7 @@ public class JobCoordinatorImpl extends BaseJobService implements IJobCoordinato
|
||||||
int recordsProcessed = outcome.getRecordsProcessed();
|
int recordsProcessed = outcome.getRecordsProcessed();
|
||||||
myJobPersistence.markWorkChunkAsCompletedAndClearData(chunkId, recordsProcessed);
|
myJobPersistence.markWorkChunkAsCompletedAndClearData(chunkId, recordsProcessed);
|
||||||
|
|
||||||
int recoveredErrorCount = dataSink.getRecoveredErrorCount();
|
int recoveredErrorCount = theDataSink.getRecoveredErrorCount();
|
||||||
if (recoveredErrorCount > 0) {
|
if (recoveredErrorCount > 0) {
|
||||||
myJobPersistence.incrementWorkChunkErrorCount(chunkId, recoveredErrorCount);
|
myJobPersistence.incrementWorkChunkErrorCount(chunkId, recoveredErrorCount);
|
||||||
}
|
}
|
||||||
|
@ -296,32 +281,40 @@ public class JobCoordinatorImpl extends BaseJobService implements IJobCoordinato
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
executeStep(chunk, jobDefinitionId, jobDefinitionVersion, definition, targetStep, nextStep, targetStepId, firstStep, instance, instanceId);
|
executeStep(chunk, jobDefinitionId, jobDefinitionVersion, definition, targetStep, nextStep, targetStepId, firstStep, instance);
|
||||||
}
|
}
|
||||||
|
|
||||||
@SuppressWarnings("unchecked")
|
@SuppressWarnings("unchecked")
|
||||||
private <PT extends IModelJson, IT extends IModelJson, OT extends IModelJson> void executeStep(WorkChunk chunk, String jobDefinitionId, int jobDefinitionVersion, JobDefinition<PT> definition, JobDefinitionStep<PT, IT, OT> theStep, JobDefinitionStep<PT, OT, ?> theSubsequentStep, String targetStepId, boolean firstStep, JobInstance instance, String instanceId) {
|
private <PT extends IModelJson, IT extends IModelJson, OT extends IModelJson> void executeStep(WorkChunk theWorkChunk, String theJobDefinitionId, int theJobDefinitionVersion, JobDefinition<PT> theDefinition, JobDefinitionStep<PT, IT, OT> theStep, JobDefinitionStep<PT, OT, ?> theSubsequentStep, String theTargetStepId, boolean theFirstStep, JobInstance theInstance) {
|
||||||
PT parameters = (PT) instance.getParameters(definition.getParametersType());
|
String instanceId = theInstance.getInstanceId();
|
||||||
|
PT parameters = theInstance.getParameters(theDefinition.getParametersType());
|
||||||
IJobStepWorker<PT, IT, OT> worker = theStep.getJobStepWorker();
|
IJobStepWorker<PT, IT, OT> worker = theStep.getJobStepWorker();
|
||||||
|
|
||||||
BaseDataSink<OT> dataSink;
|
BaseDataSink<OT> dataSink;
|
||||||
if (theSubsequentStep != null) {
|
boolean finalStep = theSubsequentStep == null;
|
||||||
dataSink = new JobDataSink<>(myBatchJobSender, myJobPersistence, jobDefinitionId, jobDefinitionVersion, theSubsequentStep, instanceId, theStep.getStepId());
|
if (!finalStep) {
|
||||||
|
dataSink = new JobDataSink<>(myBatchJobSender, myJobPersistence, theJobDefinitionId, theJobDefinitionVersion, theSubsequentStep, instanceId, theStep.getStepId(), theDefinition.isGatedExecution());
|
||||||
} else {
|
} else {
|
||||||
dataSink = (BaseDataSink<OT>) new FinalStepDataSink(jobDefinitionId, instanceId, theStep.getStepId());
|
dataSink = (BaseDataSink<OT>) new FinalStepDataSink(theJobDefinitionId, instanceId, theStep.getStepId());
|
||||||
}
|
}
|
||||||
|
|
||||||
Class<IT> inputType = theStep.getInputType();
|
Class<IT> inputType = theStep.getInputType();
|
||||||
boolean success = executeStep(chunk, jobDefinitionId, targetStepId, inputType, parameters, worker, dataSink);
|
boolean success = executeStep(theWorkChunk, theJobDefinitionId, theTargetStepId, inputType, parameters, worker, dataSink);
|
||||||
if (!success) {
|
if (!success) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
int workChunkCount = dataSink.getWorkChunkCount();
|
int workChunkCount = dataSink.getWorkChunkCount();
|
||||||
if (firstStep && workChunkCount == 0) {
|
if (theFirstStep && workChunkCount == 0) {
|
||||||
ourLog.info("First step of job instance {} produced no work chunks, marking as completed", instanceId);
|
ourLog.info("First step of job theInstance {} produced no work chunks, marking as completed", instanceId);
|
||||||
myJobPersistence.markInstanceAsCompleted(instanceId);
|
myJobPersistence.markInstanceAsCompleted(instanceId);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (theDefinition.isGatedExecution() && theFirstStep) {
|
||||||
|
theInstance.setCurrentGatedStepId(theTargetStepId);
|
||||||
|
myJobPersistence.updateInstance(theInstance);
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private JobDefinition<?> getDefinitionOrThrowException(String jobDefinitionId, int jobDefinitionVersion) {
|
private JobDefinition<?> getDefinitionOrThrowException(String jobDefinitionId, int jobDefinitionVersion) {
|
||||||
|
|
|
@ -36,14 +36,16 @@ class JobDataSink<OT extends IModelJson> extends BaseDataSink<OT> {
|
||||||
private final int myJobDefinitionVersion;
|
private final int myJobDefinitionVersion;
|
||||||
private final JobDefinitionStep<?, ?, ?> myTargetStep;
|
private final JobDefinitionStep<?, ?, ?> myTargetStep;
|
||||||
private final AtomicInteger myChunkCounter = new AtomicInteger(0);
|
private final AtomicInteger myChunkCounter = new AtomicInteger(0);
|
||||||
|
private final boolean myGatedExecution;
|
||||||
|
|
||||||
JobDataSink(BatchJobSender theBatchJobSender, IJobPersistence theJobPersistence, String theJobDefinitionId, int theJobDefinitionVersion, JobDefinitionStep<?, ?, ?> theTargetStep, String theInstanceId, String theCurrentStepId) {
|
JobDataSink(BatchJobSender theBatchJobSender, IJobPersistence theJobPersistence, String theJobDefinitionId, int theJobDefinitionVersion, JobDefinitionStep<?, ?, ?> theTargetStep, String theInstanceId, String theCurrentStepId, boolean theGatedExecution) {
|
||||||
super(theInstanceId, theCurrentStepId);
|
super(theInstanceId, theCurrentStepId);
|
||||||
myBatchJobSender = theBatchJobSender;
|
myBatchJobSender = theBatchJobSender;
|
||||||
myJobPersistence = theJobPersistence;
|
myJobPersistence = theJobPersistence;
|
||||||
myJobDefinitionId = theJobDefinitionId;
|
myJobDefinitionId = theJobDefinitionId;
|
||||||
myJobDefinitionVersion = theJobDefinitionVersion;
|
myJobDefinitionVersion = theJobDefinitionVersion;
|
||||||
myTargetStep = theTargetStep;
|
myTargetStep = theTargetStep;
|
||||||
|
myGatedExecution = theGatedExecution;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -59,8 +61,10 @@ class JobDataSink<OT extends IModelJson> extends BaseDataSink<OT> {
|
||||||
BatchWorkChunk batchWorkChunk = new BatchWorkChunk(jobDefinitionId, jobDefinitionVersion, targetStepId, instanceId, sequence, dataValueString);
|
BatchWorkChunk batchWorkChunk = new BatchWorkChunk(jobDefinitionId, jobDefinitionVersion, targetStepId, instanceId, sequence, dataValueString);
|
||||||
String chunkId = myJobPersistence.storeWorkChunk(batchWorkChunk);
|
String chunkId = myJobPersistence.storeWorkChunk(batchWorkChunk);
|
||||||
|
|
||||||
JobWorkNotification workNotification = new JobWorkNotification(jobDefinitionId, jobDefinitionVersion, instanceId, targetStepId, chunkId);
|
if (!myGatedExecution) {
|
||||||
myBatchJobSender.sendWorkChannelMessage(workNotification);
|
JobWorkNotification workNotification = new JobWorkNotification(jobDefinitionId, jobDefinitionVersion, instanceId, targetStepId, chunkId);
|
||||||
|
myBatchJobSender.sendWorkChannelMessage(workNotification);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
|
@ -20,15 +20,22 @@ package ca.uhn.fhir.batch2.impl;
|
||||||
* #L%
|
* #L%
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import ca.uhn.fhir.batch2.api.IJobCleanerService;
|
import ca.uhn.fhir.batch2.api.IJobCompletionHandler;
|
||||||
|
import ca.uhn.fhir.batch2.api.IJobMaintenanceService;
|
||||||
import ca.uhn.fhir.batch2.api.IJobPersistence;
|
import ca.uhn.fhir.batch2.api.IJobPersistence;
|
||||||
|
import ca.uhn.fhir.batch2.api.JobCompletionDetails;
|
||||||
|
import ca.uhn.fhir.batch2.model.JobDefinition;
|
||||||
import ca.uhn.fhir.batch2.model.JobInstance;
|
import ca.uhn.fhir.batch2.model.JobInstance;
|
||||||
|
import ca.uhn.fhir.batch2.model.JobWorkNotification;
|
||||||
import ca.uhn.fhir.batch2.model.StatusEnum;
|
import ca.uhn.fhir.batch2.model.StatusEnum;
|
||||||
import ca.uhn.fhir.batch2.model.WorkChunk;
|
import ca.uhn.fhir.batch2.model.WorkChunk;
|
||||||
import ca.uhn.fhir.jpa.model.sched.HapiJob;
|
import ca.uhn.fhir.jpa.model.sched.HapiJob;
|
||||||
import ca.uhn.fhir.jpa.model.sched.ISchedulerService;
|
import ca.uhn.fhir.jpa.model.sched.ISchedulerService;
|
||||||
import ca.uhn.fhir.jpa.model.sched.ScheduledJobDefinition;
|
import ca.uhn.fhir.jpa.model.sched.ScheduledJobDefinition;
|
||||||
|
import ca.uhn.fhir.model.api.IModelJson;
|
||||||
import ca.uhn.fhir.util.StopWatch;
|
import ca.uhn.fhir.util.StopWatch;
|
||||||
|
import com.google.common.collect.ArrayListMultimap;
|
||||||
|
import com.google.common.collect.Multimap;
|
||||||
import org.apache.commons.lang3.Validate;
|
import org.apache.commons.lang3.Validate;
|
||||||
import org.apache.commons.lang3.time.DateUtils;
|
import org.apache.commons.lang3.time.DateUtils;
|
||||||
import org.quartz.JobExecutionContext;
|
import org.quartz.JobExecutionContext;
|
||||||
|
@ -36,16 +43,27 @@ import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
import org.springframework.beans.factory.annotation.Autowired;
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
|
|
||||||
|
import javax.annotation.Nonnull;
|
||||||
import javax.annotation.PostConstruct;
|
import javax.annotation.PostConstruct;
|
||||||
|
import java.util.Collection;
|
||||||
import java.util.Date;
|
import java.util.Date;
|
||||||
|
import java.util.EnumSet;
|
||||||
import java.util.HashSet;
|
import java.util.HashSet;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
import java.util.concurrent.TimeUnit;
|
import java.util.concurrent.TimeUnit;
|
||||||
|
import java.util.stream.Collectors;
|
||||||
|
|
||||||
|
import static java.util.Collections.emptyList;
|
||||||
|
import static org.apache.commons.lang3.ObjectUtils.defaultIfNull;
|
||||||
|
import static org.apache.commons.lang3.StringUtils.isBlank;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This class performs regular polls of the stored jobs in order to
|
* This class performs regular polls of the stored jobs in order to
|
||||||
* calculate statistics and delete expired tasks. This class does
|
* perform maintenance. This includes two major functions.
|
||||||
|
*
|
||||||
|
* <p>
|
||||||
|
* First, we calculate statistics and delete expired tasks. This class does
|
||||||
* the following things:
|
* the following things:
|
||||||
* <ul>
|
* <ul>
|
||||||
* <li>For instances that are IN_PROGRESS, calculates throughput and percent complete</li>
|
* <li>For instances that are IN_PROGRESS, calculates throughput and percent complete</li>
|
||||||
|
@ -55,45 +73,60 @@ import java.util.concurrent.TimeUnit;
|
||||||
* <li>For instances that are IN_PROGRESS with an error message set where no chunks are ERRORED or FAILED, clears the error message in the instance (meaning presumably there was an error but it cleared)</li>
|
* <li>For instances that are IN_PROGRESS with an error message set where no chunks are ERRORED or FAILED, clears the error message in the instance (meaning presumably there was an error but it cleared)</li>
|
||||||
* <li>For instances that are COMPLETE or FAILED and are old, delete them entirely</li>
|
* <li>For instances that are COMPLETE or FAILED and are old, delete them entirely</li>
|
||||||
* </ul>
|
* </ul>
|
||||||
|
* </p>
|
||||||
|
*
|
||||||
|
* <p>
|
||||||
|
* Second, we check for any job instances where the job is configured to
|
||||||
|
* have gated execution. For these instances, we check if the current step
|
||||||
|
* is complete (all chunks are in COMPLETE status) and trigger the next step.
|
||||||
|
* </p>
|
||||||
*/
|
*/
|
||||||
public class JobCleanerServiceImpl extends BaseJobService implements IJobCleanerService {
|
public class JobMaintenanceServiceImpl extends BaseJobService implements IJobMaintenanceService {
|
||||||
|
|
||||||
public static final int INSTANCES_PER_PASS = 100;
|
public static final int INSTANCES_PER_PASS = 100;
|
||||||
public static final long PURGE_THRESHOLD = 7L * DateUtils.MILLIS_PER_DAY;
|
public static final long PURGE_THRESHOLD = 7L * DateUtils.MILLIS_PER_DAY;
|
||||||
private static final Logger ourLog = LoggerFactory.getLogger(JobCleanerServiceImpl.class);
|
private static final Logger ourLog = LoggerFactory.getLogger(JobMaintenanceServiceImpl.class);
|
||||||
private final ISchedulerService mySchedulerService;
|
private final ISchedulerService mySchedulerService;
|
||||||
|
private final JobDefinitionRegistry myJobDefinitionRegistry;
|
||||||
|
private final BatchJobSender myBatchJobSender;
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Constructor
|
* Constructor
|
||||||
*/
|
*/
|
||||||
public JobCleanerServiceImpl(ISchedulerService theSchedulerService, IJobPersistence theJobPersistence) {
|
public JobMaintenanceServiceImpl(ISchedulerService theSchedulerService, IJobPersistence theJobPersistence, JobDefinitionRegistry theJobDefinitionRegistry, BatchJobSender theBatchJobSender) {
|
||||||
super(theJobPersistence);
|
super(theJobPersistence);
|
||||||
Validate.notNull(theSchedulerService);
|
Validate.notNull(theSchedulerService);
|
||||||
|
Validate.notNull(theJobDefinitionRegistry);
|
||||||
|
Validate.notNull(theBatchJobSender);
|
||||||
|
|
||||||
mySchedulerService = theSchedulerService;
|
mySchedulerService = theSchedulerService;
|
||||||
|
myJobDefinitionRegistry = theJobDefinitionRegistry;
|
||||||
|
myBatchJobSender = theBatchJobSender;
|
||||||
}
|
}
|
||||||
|
|
||||||
@PostConstruct
|
@PostConstruct
|
||||||
public void start() {
|
public void start() {
|
||||||
ScheduledJobDefinition jobDefinition = new ScheduledJobDefinition();
|
ScheduledJobDefinition jobDefinition = new ScheduledJobDefinition();
|
||||||
jobDefinition.setId(JobCleanerScheduledJob.class.getName());
|
jobDefinition.setId(JobMaintenanceScheduledJob.class.getName());
|
||||||
jobDefinition.setJobClass(JobCleanerScheduledJob.class);
|
jobDefinition.setJobClass(JobMaintenanceScheduledJob.class);
|
||||||
mySchedulerService.scheduleClusteredJob(DateUtils.MILLIS_PER_MINUTE, jobDefinition);
|
mySchedulerService.scheduleClusteredJob(DateUtils.MILLIS_PER_MINUTE, jobDefinition);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void runCleanupPass() {
|
public void runMaintenancePass() {
|
||||||
|
|
||||||
// NB: If you add any new logic, update the class javadoc
|
// NB: If you add any new logic, update the class javadoc
|
||||||
|
|
||||||
Set<String> processedInstanceIds = new HashSet<>();
|
Set<String> processedInstanceIds = new HashSet<>();
|
||||||
|
JobChunkProgressAccumulator progressAccumulator = new JobChunkProgressAccumulator();
|
||||||
for (int page = 0; ; page++) {
|
for (int page = 0; ; page++) {
|
||||||
List<JobInstance> instances = myJobPersistence.fetchInstances(INSTANCES_PER_PASS, page);
|
List<JobInstance> instances = myJobPersistence.fetchInstances(INSTANCES_PER_PASS, page);
|
||||||
|
|
||||||
for (JobInstance instance : instances) {
|
for (JobInstance instance : instances) {
|
||||||
if (processedInstanceIds.add(instance.getInstanceId())) {
|
if (processedInstanceIds.add(instance.getInstanceId())) {
|
||||||
cleanupInstance(instance);
|
cleanupInstance(instance, progressAccumulator);
|
||||||
|
triggerGatedExecutions(instance, progressAccumulator);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -103,13 +136,13 @@ public class JobCleanerServiceImpl extends BaseJobService implements IJobCleaner
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void cleanupInstance(JobInstance theInstance) {
|
private void cleanupInstance(JobInstance theInstance, JobChunkProgressAccumulator theProgressAccumulator) {
|
||||||
switch (theInstance.getStatus()) {
|
switch (theInstance.getStatus()) {
|
||||||
case QUEUED:
|
case QUEUED:
|
||||||
break;
|
break;
|
||||||
case IN_PROGRESS:
|
case IN_PROGRESS:
|
||||||
case ERRORED:
|
case ERRORED:
|
||||||
calculateInstanceProgress(theInstance);
|
calculateInstanceProgress(theInstance, theProgressAccumulator);
|
||||||
break;
|
break;
|
||||||
case COMPLETED:
|
case COMPLETED:
|
||||||
case FAILED:
|
case FAILED:
|
||||||
|
@ -132,7 +165,7 @@ public class JobCleanerServiceImpl extends BaseJobService implements IJobCleaner
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private void calculateInstanceProgress(JobInstance theInstance) {
|
private void calculateInstanceProgress(JobInstance theInstance, JobChunkProgressAccumulator theProgressAccumulator) {
|
||||||
int resourcesProcessed = 0;
|
int resourcesProcessed = 0;
|
||||||
int incompleteChunkCount = 0;
|
int incompleteChunkCount = 0;
|
||||||
int completeChunkCount = 0;
|
int completeChunkCount = 0;
|
||||||
|
@ -146,6 +179,8 @@ public class JobCleanerServiceImpl extends BaseJobService implements IJobCleaner
|
||||||
List<WorkChunk> chunks = myJobPersistence.fetchWorkChunksWithoutData(theInstance.getInstanceId(), INSTANCES_PER_PASS, page);
|
List<WorkChunk> chunks = myJobPersistence.fetchWorkChunksWithoutData(theInstance.getInstanceId(), INSTANCES_PER_PASS, page);
|
||||||
|
|
||||||
for (WorkChunk chunk : chunks) {
|
for (WorkChunk chunk : chunks) {
|
||||||
|
theProgressAccumulator.addChunk(chunk.getInstanceId(), chunk.getId(), chunk.getTargetStepId(), chunk.getStatus());
|
||||||
|
|
||||||
errorCountForAllStatuses += chunk.getErrorCount();
|
errorCountForAllStatuses += chunk.getErrorCount();
|
||||||
|
|
||||||
if (chunk.getRecordsProcessed() != null) {
|
if (chunk.getRecordsProcessed() != null) {
|
||||||
|
@ -201,7 +236,12 @@ public class JobCleanerServiceImpl extends BaseJobService implements IJobCleaner
|
||||||
|
|
||||||
changedStatus = false;
|
changedStatus = false;
|
||||||
if (incompleteChunkCount == 0 && erroredChunkCount == 0 && failedChunkCount == 0) {
|
if (incompleteChunkCount == 0 && erroredChunkCount == 0 && failedChunkCount == 0) {
|
||||||
changedStatus |= updateInstanceStatus(theInstance, StatusEnum.COMPLETED);
|
boolean completed = updateInstanceStatus(theInstance, StatusEnum.COMPLETED);
|
||||||
|
if (completed) {
|
||||||
|
JobDefinition<?> definition = myJobDefinitionRegistry.getJobDefinition(theInstance.getJobDefinitionId(), theInstance.getJobDefinitionVersion()).orElseThrow(() -> new IllegalStateException("Unknown job " + theInstance.getJobDefinitionId() + "/" + theInstance.getJobDefinitionVersion()));
|
||||||
|
invokeJobCompletionHandler(theInstance, definition);
|
||||||
|
}
|
||||||
|
changedStatus |= completed;
|
||||||
}
|
}
|
||||||
if (erroredChunkCount > 0) {
|
if (erroredChunkCount > 0) {
|
||||||
changedStatus |= updateInstanceStatus(theInstance, StatusEnum.ERRORED);
|
changedStatus |= updateInstanceStatus(theInstance, StatusEnum.ERRORED);
|
||||||
|
@ -246,6 +286,17 @@ public class JobCleanerServiceImpl extends BaseJobService implements IJobCleaner
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private <PT extends IModelJson> void invokeJobCompletionHandler(JobInstance theInstance, JobDefinition<PT> definition) {
|
||||||
|
IJobCompletionHandler<PT> completionHandler = definition.getCompletionHandler();
|
||||||
|
if (completionHandler != null) {
|
||||||
|
|
||||||
|
String instanceId = theInstance.getInstanceId();
|
||||||
|
PT jobParameters = theInstance.getParameters(definition.getParametersType());
|
||||||
|
JobCompletionDetails<PT> completionDetails = new JobCompletionDetails<>(jobParameters, instanceId);
|
||||||
|
completionHandler.jobComplete(completionDetails);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private boolean updateInstanceStatus(JobInstance theInstance, StatusEnum newStatus) {
|
private boolean updateInstanceStatus(JobInstance theInstance, StatusEnum newStatus) {
|
||||||
if (theInstance.getStatus() != newStatus) {
|
if (theInstance.getStatus() != newStatus) {
|
||||||
ourLog.info("Marking job instance {} of type {} as {}", theInstance.getInstanceId(), theInstance.getJobDefinitionId(), newStatus);
|
ourLog.info("Marking job instance {} of type {} as {}", theInstance.getInstanceId(), theInstance.getJobDefinitionId(), newStatus);
|
||||||
|
@ -255,16 +306,109 @@ public class JobCleanerServiceImpl extends BaseJobService implements IJobCleaner
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void triggerGatedExecutions(JobInstance theInstance, JobChunkProgressAccumulator theProgressAccumulator) {
|
||||||
|
if (!theInstance.isRunning()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
public static class JobCleanerScheduledJob implements HapiJob {
|
String jobDefinitionId = theInstance.getJobDefinitionId();
|
||||||
|
int jobDefinitionVersion = theInstance.getJobDefinitionVersion();
|
||||||
|
String instanceId = theInstance.getInstanceId();
|
||||||
|
|
||||||
|
JobDefinition<?> definition = myJobDefinitionRegistry.getJobDefinition(jobDefinitionId, jobDefinitionVersion).orElseThrow(() -> new IllegalStateException("Unknown job definition: " + jobDefinitionId + " " + jobDefinitionVersion));
|
||||||
|
if (!definition.isGatedExecution()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
String currentStepId = theInstance.getCurrentGatedStepId();
|
||||||
|
if (isBlank(currentStepId)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (definition.isLastStep(currentStepId)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
int incompleteChunks = theProgressAccumulator.countChunksWithStatus(instanceId, currentStepId, StatusEnum.getIncompleteStatuses());
|
||||||
|
if (incompleteChunks == 0) {
|
||||||
|
|
||||||
|
int currentStepIndex = definition.getStepIndex(currentStepId);
|
||||||
|
String nextStepId = definition.getSteps().get(currentStepIndex + 1).getStepId();
|
||||||
|
|
||||||
|
ourLog.info("All processing is complete for gated execution of instance {} step {}. Proceeding to step {}", instanceId, currentStepId, nextStepId);
|
||||||
|
List<String> chunksForNextStep = theProgressAccumulator.getChunkIdsWithStatus(instanceId, nextStepId, EnumSet.of(StatusEnum.QUEUED));
|
||||||
|
for (String nextChunkId : chunksForNextStep) {
|
||||||
|
JobWorkNotification workNotification = new JobWorkNotification(jobDefinitionId, jobDefinitionVersion, instanceId, nextStepId, nextChunkId);
|
||||||
|
myBatchJobSender.sendWorkChannelMessage(workNotification);
|
||||||
|
}
|
||||||
|
|
||||||
|
theInstance.setCurrentGatedStepId(nextStepId);
|
||||||
|
myJobPersistence.updateInstance(theInstance);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public static class JobMaintenanceScheduledJob implements HapiJob {
|
||||||
@Autowired
|
@Autowired
|
||||||
private IJobCleanerService myTarget;
|
private IJobMaintenanceService myTarget;
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void execute(JobExecutionContext theContext) {
|
public void execute(JobExecutionContext theContext) {
|
||||||
myTarget.runCleanupPass();
|
myTarget.runMaintenancePass();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* While performing cleanup, the cleanup job loads all of the known
|
||||||
|
* work chunks to examine their status. This bean collects the counts that
|
||||||
|
* are found, so that they can be reused for maintenance jobs without
|
||||||
|
* needing to hit the database a second time.
|
||||||
|
*/
|
||||||
|
private static class JobChunkProgressAccumulator {
|
||||||
|
|
||||||
|
private final Set<String> myConsumedInstanceAndChunkIds = new HashSet<>();
|
||||||
|
private final Multimap<String, ChunkStatusCountKey> myInstanceIdToChunkStatuses = ArrayListMultimap.create();
|
||||||
|
|
||||||
|
public void addChunk(String theInstanceId, String theChunkId, String theStepId, StatusEnum theStatus) {
|
||||||
|
// Note: If chunks are being written while we're executing, we may see the same chunk twice. This
|
||||||
|
// check avoids adding it twice.
|
||||||
|
if (myConsumedInstanceAndChunkIds.add(theInstanceId + " " + theChunkId)) {
|
||||||
|
myInstanceIdToChunkStatuses.put(theInstanceId, new ChunkStatusCountKey(theChunkId, theStepId, theStatus));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public int countChunksWithStatus(String theInstanceId, String theStepId, Set<StatusEnum> theStatuses) {
|
||||||
|
return getChunkIdsWithStatus(theInstanceId, theStepId, theStatuses).size();
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<String> getChunkIdsWithStatus(String theInstanceId, String theStepId, Set<StatusEnum> theStatuses) {
|
||||||
|
return getChunkStatuses(theInstanceId).stream().filter(t -> t.myStepId.equals(theStepId)).filter(t -> theStatuses.contains(t.myStatus)).map(t -> t.myChunkId).collect(Collectors.toList());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nonnull
|
||||||
|
private Collection<ChunkStatusCountKey> getChunkStatuses(String theInstanceId) {
|
||||||
|
Collection<ChunkStatusCountKey> chunkStatuses = myInstanceIdToChunkStatuses.get(theInstanceId);
|
||||||
|
chunkStatuses = defaultIfNull(chunkStatuses, emptyList());
|
||||||
|
return chunkStatuses;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private static class ChunkStatusCountKey {
|
||||||
|
public final String myChunkId;
|
||||||
|
public final String myStepId;
|
||||||
|
public final StatusEnum myStatus;
|
||||||
|
|
||||||
|
private ChunkStatusCountKey(String theChunkId, String theStepId, StatusEnum theStatus) {
|
||||||
|
myChunkId = theChunkId;
|
||||||
|
myStepId = theStepId;
|
||||||
|
myStatus = theStatus;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
|
@ -22,8 +22,10 @@ package ca.uhn.fhir.batch2.impl;
|
||||||
|
|
||||||
import ca.uhn.fhir.batch2.api.IJobPersistence;
|
import ca.uhn.fhir.batch2.api.IJobPersistence;
|
||||||
import ca.uhn.fhir.batch2.model.JobInstance;
|
import ca.uhn.fhir.batch2.model.JobInstance;
|
||||||
|
import ca.uhn.fhir.batch2.model.StatusEnum;
|
||||||
import ca.uhn.fhir.batch2.model.WorkChunk;
|
import ca.uhn.fhir.batch2.model.WorkChunk;
|
||||||
|
|
||||||
|
import java.util.EnumSet;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Optional;
|
import java.util.Optional;
|
||||||
|
|
||||||
|
|
|
@ -20,6 +20,7 @@ package ca.uhn.fhir.batch2.model;
|
||||||
* #L%
|
* #L%
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
import ca.uhn.fhir.batch2.api.IJobCompletionHandler;
|
||||||
import ca.uhn.fhir.batch2.api.IJobParametersValidator;
|
import ca.uhn.fhir.batch2.api.IJobParametersValidator;
|
||||||
import ca.uhn.fhir.batch2.api.IJobStepWorker;
|
import ca.uhn.fhir.batch2.api.IJobStepWorker;
|
||||||
import ca.uhn.fhir.batch2.api.VoidModel;
|
import ca.uhn.fhir.batch2.api.VoidModel;
|
||||||
|
@ -31,6 +32,7 @@ import javax.annotation.Nullable;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
import java.util.stream.Collectors;
|
||||||
|
|
||||||
public class JobDefinition<PT extends IModelJson> {
|
public class JobDefinition<PT extends IModelJson> {
|
||||||
|
|
||||||
|
@ -42,11 +44,14 @@ public class JobDefinition<PT extends IModelJson> {
|
||||||
private final List<JobDefinitionStep<PT, ?, ?>> mySteps;
|
private final List<JobDefinitionStep<PT, ?, ?>> mySteps;
|
||||||
private final String myJobDescription;
|
private final String myJobDescription;
|
||||||
private final IJobParametersValidator<PT> myParametersValidator;
|
private final IJobParametersValidator<PT> myParametersValidator;
|
||||||
|
private final boolean myGatedExecution;
|
||||||
|
private final List<String> myStepIds;
|
||||||
|
private final IJobCompletionHandler<PT> myCompletionHandler;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Constructor
|
* Constructor
|
||||||
*/
|
*/
|
||||||
private JobDefinition(String theJobDefinitionId, int theJobDefinitionVersion, String theJobDescription, Class<PT> theParametersType, List<JobDefinitionStep<PT, ?, ?>> theSteps, IJobParametersValidator<PT> theParametersValidator) {
|
private JobDefinition(String theJobDefinitionId, int theJobDefinitionVersion, String theJobDescription, Class<PT> theParametersType, List<JobDefinitionStep<PT, ?, ?>> theSteps, IJobParametersValidator<PT> theParametersValidator, boolean theGatedExecution, IJobCompletionHandler<PT> theCompletionHandler) {
|
||||||
Validate.isTrue(theJobDefinitionId.length() <= ID_MAX_LENGTH, "Maximum ID length is %d", ID_MAX_LENGTH);
|
Validate.isTrue(theJobDefinitionId.length() <= ID_MAX_LENGTH, "Maximum ID length is %d", ID_MAX_LENGTH);
|
||||||
Validate.notBlank(theJobDefinitionId, "No job definition ID supplied");
|
Validate.notBlank(theJobDefinitionId, "No job definition ID supplied");
|
||||||
Validate.notBlank(theJobDescription, "No job description supplied");
|
Validate.notBlank(theJobDescription, "No job description supplied");
|
||||||
|
@ -56,8 +61,16 @@ public class JobDefinition<PT extends IModelJson> {
|
||||||
myJobDefinitionVersion = theJobDefinitionVersion;
|
myJobDefinitionVersion = theJobDefinitionVersion;
|
||||||
myJobDescription = theJobDescription;
|
myJobDescription = theJobDescription;
|
||||||
mySteps = theSteps;
|
mySteps = theSteps;
|
||||||
|
myStepIds = mySteps.stream().map(t -> t.getStepId()).collect(Collectors.toList());
|
||||||
myParametersType = theParametersType;
|
myParametersType = theParametersType;
|
||||||
myParametersValidator = theParametersValidator;
|
myParametersValidator = theParametersValidator;
|
||||||
|
myGatedExecution = theGatedExecution;
|
||||||
|
myCompletionHandler = theCompletionHandler;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nullable
|
||||||
|
public IJobCompletionHandler<PT> getCompletionHandler() {
|
||||||
|
return myCompletionHandler;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Nullable
|
@Nullable
|
||||||
|
@ -97,8 +110,18 @@ public class JobDefinition<PT extends IModelJson> {
|
||||||
return mySteps;
|
return mySteps;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static Builder<IModelJson, VoidModel> newBuilder() {
|
public boolean isGatedExecution() {
|
||||||
return new Builder<>();
|
return myGatedExecution;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getStepIndex(String theStepId) {
|
||||||
|
int retVal = myStepIds.indexOf(theStepId);
|
||||||
|
Validate.isTrue(retVal != -1);
|
||||||
|
return retVal;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isLastStep(String theStepId) {
|
||||||
|
return getStepIndex(theStepId) == (myStepIds.size() - 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static class Builder<PT extends IModelJson, NIT extends IModelJson> {
|
public static class Builder<PT extends IModelJson, NIT extends IModelJson> {
|
||||||
|
@ -111,12 +134,14 @@ public class JobDefinition<PT extends IModelJson> {
|
||||||
private Class<NIT> myNextInputType;
|
private Class<NIT> myNextInputType;
|
||||||
@Nullable
|
@Nullable
|
||||||
private IJobParametersValidator<PT> myParametersValidator;
|
private IJobParametersValidator<PT> myParametersValidator;
|
||||||
|
private boolean myGatedExecution;
|
||||||
|
private IJobCompletionHandler<PT> myCompletionHandler;
|
||||||
|
|
||||||
Builder() {
|
Builder() {
|
||||||
mySteps = new ArrayList<>();
|
mySteps = new ArrayList<>();
|
||||||
}
|
}
|
||||||
|
|
||||||
Builder(List<JobDefinitionStep<PT, ?, ?>> theSteps, String theJobDefinitionId, int theJobDefinitionVersion, String theJobDescription, Class<PT> theJobParametersType, Class<NIT> theNextInputType, IJobParametersValidator<PT> theParametersValidator) {
|
Builder(List<JobDefinitionStep<PT, ?, ?>> theSteps, String theJobDefinitionId, int theJobDefinitionVersion, String theJobDescription, Class<PT> theJobParametersType, Class<NIT> theNextInputType, IJobParametersValidator<PT> theParametersValidator, boolean theGatedExecution, IJobCompletionHandler<PT> theCompletionHandler) {
|
||||||
mySteps = theSteps;
|
mySteps = theSteps;
|
||||||
myJobDefinitionId = theJobDefinitionId;
|
myJobDefinitionId = theJobDefinitionId;
|
||||||
myJobDefinitionVersion = theJobDefinitionVersion;
|
myJobDefinitionVersion = theJobDefinitionVersion;
|
||||||
|
@ -124,6 +149,8 @@ public class JobDefinition<PT extends IModelJson> {
|
||||||
myJobParametersType = theJobParametersType;
|
myJobParametersType = theJobParametersType;
|
||||||
myNextInputType = theNextInputType;
|
myNextInputType = theNextInputType;
|
||||||
myParametersValidator = theParametersValidator;
|
myParametersValidator = theParametersValidator;
|
||||||
|
myGatedExecution = theGatedExecution;
|
||||||
|
myCompletionHandler = theCompletionHandler;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -154,7 +181,7 @@ public class JobDefinition<PT extends IModelJson> {
|
||||||
*/
|
*/
|
||||||
public <OT extends IModelJson> Builder<PT, OT> addFirstStep(String theStepId, String theStepDescription, Class<OT> theOutputType, IJobStepWorker<PT, VoidModel, OT> theStepWorker) {
|
public <OT extends IModelJson> Builder<PT, OT> addFirstStep(String theStepId, String theStepDescription, Class<OT> theOutputType, IJobStepWorker<PT, VoidModel, OT> theStepWorker) {
|
||||||
mySteps.add(new JobDefinitionStep<>(theStepId, theStepDescription, theStepWorker, VoidModel.class, theOutputType));
|
mySteps.add(new JobDefinitionStep<>(theStepId, theStepDescription, theStepWorker, VoidModel.class, theOutputType));
|
||||||
return new Builder<>(mySteps, myJobDefinitionId, myJobDefinitionVersion, myJobDescription, myJobParametersType, theOutputType, myParametersValidator);
|
return new Builder<>(mySteps, myJobDefinitionId, myJobDefinitionVersion, myJobDescription, myJobParametersType, theOutputType, myParametersValidator, myGatedExecution, myCompletionHandler);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -168,7 +195,7 @@ public class JobDefinition<PT extends IModelJson> {
|
||||||
*/
|
*/
|
||||||
public <OT extends IModelJson> Builder<PT, OT> addIntermediateStep(String theStepId, String theStepDescription, Class<OT> theOutputType, IJobStepWorker<PT, NIT, OT> theStepWorker) {
|
public <OT extends IModelJson> Builder<PT, OT> addIntermediateStep(String theStepId, String theStepDescription, Class<OT> theOutputType, IJobStepWorker<PT, NIT, OT> theStepWorker) {
|
||||||
mySteps.add(new JobDefinitionStep<>(theStepId, theStepDescription, theStepWorker, myNextInputType, theOutputType));
|
mySteps.add(new JobDefinitionStep<>(theStepId, theStepDescription, theStepWorker, myNextInputType, theOutputType));
|
||||||
return new Builder<>(mySteps, myJobDefinitionId, myJobDefinitionVersion, myJobDescription, myJobParametersType, theOutputType, myParametersValidator);
|
return new Builder<>(mySteps, myJobDefinitionId, myJobDefinitionVersion, myJobDescription, myJobParametersType, theOutputType, myParametersValidator, myGatedExecution, myCompletionHandler);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -182,12 +209,12 @@ public class JobDefinition<PT extends IModelJson> {
|
||||||
*/
|
*/
|
||||||
public Builder<PT, VoidModel> addLastStep(String theStepId, String theStepDescription, IJobStepWorker<PT, NIT, VoidModel> theStepWorker) {
|
public Builder<PT, VoidModel> addLastStep(String theStepId, String theStepDescription, IJobStepWorker<PT, NIT, VoidModel> theStepWorker) {
|
||||||
mySteps.add(new JobDefinitionStep<>(theStepId, theStepDescription, theStepWorker, myNextInputType, VoidModel.class));
|
mySteps.add(new JobDefinitionStep<>(theStepId, theStepDescription, theStepWorker, myNextInputType, VoidModel.class));
|
||||||
return new Builder<>(mySteps, myJobDefinitionId, myJobDefinitionVersion, myJobDescription, myJobParametersType, VoidModel.class, myParametersValidator);
|
return new Builder<>(mySteps, myJobDefinitionId, myJobDefinitionVersion, myJobDescription, myJobParametersType, VoidModel.class, myParametersValidator, myGatedExecution, myCompletionHandler);
|
||||||
}
|
}
|
||||||
|
|
||||||
public JobDefinition<PT> build() {
|
public JobDefinition<PT> build() {
|
||||||
Validate.notNull(myJobParametersType, "No job parameters type was supplied");
|
Validate.notNull(myJobParametersType, "No job parameters type was supplied");
|
||||||
return new JobDefinition<>(myJobDefinitionId, myJobDefinitionVersion, myJobDescription, myJobParametersType, Collections.unmodifiableList(mySteps), myParametersValidator);
|
return new JobDefinition<>(myJobDefinitionId, myJobDefinitionVersion, myJobDescription, myJobParametersType, Collections.unmodifiableList(mySteps), myParametersValidator, myGatedExecution, myCompletionHandler);
|
||||||
}
|
}
|
||||||
|
|
||||||
public Builder<PT, NIT> setJobDescription(String theJobDescription) {
|
public Builder<PT, NIT> setJobDescription(String theJobDescription) {
|
||||||
|
@ -233,12 +260,58 @@ public class JobDefinition<PT extends IModelJson> {
|
||||||
* @param theParametersValidator The validator (must not be null. Do not call this method at all if you do not want a parameters validator).
|
* @param theParametersValidator The validator (must not be null. Do not call this method at all if you do not want a parameters validator).
|
||||||
*/
|
*/
|
||||||
@SuppressWarnings("unchecked")
|
@SuppressWarnings("unchecked")
|
||||||
public <NPT extends IModelJson> Builder<PT, NIT> setParametersValidator(@Nonnull IJobParametersValidator<PT> theParametersValidator) {
|
public Builder<PT, NIT> setParametersValidator(@Nonnull IJobParametersValidator<PT> theParametersValidator) {
|
||||||
Validate.notNull(theParametersValidator, "theParametersValidator must not be null");
|
Validate.notNull(theParametersValidator, "theParametersValidator must not be null");
|
||||||
Validate.isTrue(myParametersValidator == null, "Can not supply multiple parameters validators. Already have: %s", myParametersValidator);
|
Validate.isTrue(myParametersValidator == null, "Can not supply multiple parameters validators. Already have: %s", myParametersValidator);
|
||||||
myParametersValidator = theParametersValidator;
|
myParametersValidator = theParametersValidator;
|
||||||
return (Builder<PT, NIT>) this;
|
return this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* If this is set, the framework will wait for all work chunks to be
|
||||||
|
* processed for an individual step before moving on to beginning
|
||||||
|
* processing on the next step. Otherwise, processing on subsequent
|
||||||
|
* steps may begin as soon as any data has been produced.
|
||||||
|
* <p>
|
||||||
|
* This is useful in a few cases:
|
||||||
|
* <ul>
|
||||||
|
* <li>
|
||||||
|
* If there are potential constraint issues, e.g. data being
|
||||||
|
* written by the third step depends on all data from the
|
||||||
|
* second step already being written
|
||||||
|
* </li>
|
||||||
|
* <li>
|
||||||
|
* If multiple steps require expensive database queries, it may
|
||||||
|
* reduce the chances of timeouts to ensure that they are run
|
||||||
|
* discretely.
|
||||||
|
* </li>
|
||||||
|
* </ul>
|
||||||
|
* </p>
|
||||||
|
* <p>
|
||||||
|
* Setting this mode means the job may take longer, since it will
|
||||||
|
* rely on a polling mechanism to determine that one step is
|
||||||
|
* complete before beginning any processing for the next step.
|
||||||
|
* </p>
|
||||||
|
*/
|
||||||
|
public Builder<PT, NIT> gatedExecution() {
|
||||||
|
myGatedExecution = true;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Supplies an optional callback that will be invoked when the job is complete
|
||||||
|
*/
|
||||||
|
public Builder<PT, NIT> completionHandler(IJobCompletionHandler<PT> theCompletionHandler) {
|
||||||
|
Validate.isTrue(myCompletionHandler == null, "Can not supply multiple completion handlers");
|
||||||
|
myCompletionHandler = theCompletionHandler;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public static Builder<IModelJson, VoidModel> newBuilder() {
|
||||||
|
return new Builder<>();
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -74,13 +74,12 @@ public class JobInstance extends JobInstanceStartRequest implements IModelJson {
|
||||||
|
|
||||||
@JsonProperty(value = "progress", access = JsonProperty.Access.READ_ONLY)
|
@JsonProperty(value = "progress", access = JsonProperty.Access.READ_ONLY)
|
||||||
private double myProgress;
|
private double myProgress;
|
||||||
|
@JsonProperty(value = "currentGatedStepId", access = JsonProperty.Access.READ_ONLY)
|
||||||
|
private String myCurrentGatedStepId;
|
||||||
@JsonProperty(value = "errorMessage", access = JsonProperty.Access.READ_ONLY)
|
@JsonProperty(value = "errorMessage", access = JsonProperty.Access.READ_ONLY)
|
||||||
private String myErrorMessage;
|
private String myErrorMessage;
|
||||||
|
|
||||||
@JsonProperty(value = "errorCount", access = JsonProperty.Access.READ_ONLY)
|
@JsonProperty(value = "errorCount", access = JsonProperty.Access.READ_ONLY)
|
||||||
private int myErrorCount;
|
private int myErrorCount;
|
||||||
|
|
||||||
@JsonProperty(value = "estimatedCompletion", access = JsonProperty.Access.READ_ONLY)
|
@JsonProperty(value = "estimatedCompletion", access = JsonProperty.Access.READ_ONLY)
|
||||||
private String myEstimatedTimeRemaining;
|
private String myEstimatedTimeRemaining;
|
||||||
|
|
||||||
|
@ -111,6 +110,15 @@ public class JobInstance extends JobInstanceStartRequest implements IModelJson {
|
||||||
setStatus(theJobInstance.getStatus());
|
setStatus(theJobInstance.getStatus());
|
||||||
setTotalElapsedMillis(theJobInstance.getTotalElapsedMillis());
|
setTotalElapsedMillis(theJobInstance.getTotalElapsedMillis());
|
||||||
setWorkChunksPurged(theJobInstance.isWorkChunksPurged());
|
setWorkChunksPurged(theJobInstance.isWorkChunksPurged());
|
||||||
|
setCurrentGatedStepId(theJobInstance.getCurrentGatedStepId());
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getCurrentGatedStepId() {
|
||||||
|
return myCurrentGatedStepId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setCurrentGatedStepId(String theCurrentGatedStepId) {
|
||||||
|
myCurrentGatedStepId = theCurrentGatedStepId;
|
||||||
}
|
}
|
||||||
|
|
||||||
public int getErrorCount() {
|
public int getErrorCount() {
|
||||||
|
@ -258,4 +266,11 @@ public class JobInstance extends JobInstanceStartRequest implements IModelJson {
|
||||||
.append("estimatedTimeRemaining", myEstimatedTimeRemaining)
|
.append("estimatedTimeRemaining", myEstimatedTimeRemaining)
|
||||||
.toString();
|
.toString();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns true if the job instance is in {@link StatusEnum#IN_PROGRESS} and is not cancelled
|
||||||
|
*/
|
||||||
|
public boolean isRunning() {
|
||||||
|
return getStatus() == StatusEnum.IN_PROGRESS && !isCancelled();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -20,33 +20,63 @@ package ca.uhn.fhir.batch2.model;
|
||||||
* #L%
|
* #L%
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.EnumSet;
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
public enum StatusEnum {
|
public enum StatusEnum {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Task is waiting to execute and should begin with no intervention required.
|
* Task is waiting to execute and should begin with no intervention required.
|
||||||
*/
|
*/
|
||||||
QUEUED,
|
QUEUED(true),
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Task is current executing
|
* Task is current executing
|
||||||
*/
|
*/
|
||||||
IN_PROGRESS,
|
IN_PROGRESS(true),
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Task completed successfully
|
* Task completed successfully
|
||||||
*/
|
*/
|
||||||
COMPLETED,
|
COMPLETED(false),
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Task execution resulted in an error but the error may be transient (or transient status is unknown).
|
* Task execution resulted in an error but the error may be transient (or transient status is unknown).
|
||||||
* Retrying may result in success.
|
* Retrying may result in success.
|
||||||
*/
|
*/
|
||||||
ERRORED,
|
ERRORED(true),
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Task has failed and is known to be unrecoverable. There is no reason to believe that retrying will
|
* Task has failed and is known to be unrecoverable. There is no reason to believe that retrying will
|
||||||
* result in a different outcome.
|
* result in a different outcome.
|
||||||
*/
|
*/
|
||||||
FAILED
|
FAILED(true);
|
||||||
|
|
||||||
|
private final boolean myIncomplete;
|
||||||
|
private static Set<StatusEnum> ourIncompleteStatuses;
|
||||||
|
|
||||||
|
StatusEnum(boolean theIncomplete) {
|
||||||
|
myIncomplete = theIncomplete;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Statuses that represent a job that has not yet completed. I.e.
|
||||||
|
* all statuses except {@link #COMPLETED}
|
||||||
|
*/
|
||||||
|
public static Set<StatusEnum> getIncompleteStatuses() {
|
||||||
|
Set<StatusEnum> retVal = ourIncompleteStatuses;
|
||||||
|
if (retVal == null) {
|
||||||
|
EnumSet<StatusEnum> set = EnumSet.noneOf(StatusEnum.class);
|
||||||
|
for (StatusEnum next : values()) {
|
||||||
|
if (next.myIncomplete) {
|
||||||
|
set.add(next);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
ourIncompleteStatuses = Collections.unmodifiableSet(set);
|
||||||
|
retVal = ourIncompleteStatuses;
|
||||||
|
}
|
||||||
|
return retVal;
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,74 @@
|
||||||
|
package ca.uhn.fhir.batch2.impl;
|
||||||
|
|
||||||
|
import ca.uhn.fhir.batch2.api.IJobStepWorker;
|
||||||
|
import ca.uhn.fhir.batch2.api.VoidModel;
|
||||||
|
import ca.uhn.fhir.batch2.model.JobDefinition;
|
||||||
|
import ca.uhn.fhir.batch2.model.JobInstance;
|
||||||
|
import ca.uhn.fhir.batch2.model.StatusEnum;
|
||||||
|
import org.junit.jupiter.api.extension.ExtendWith;
|
||||||
|
import org.mockito.Mock;
|
||||||
|
import org.mockito.junit.jupiter.MockitoExtension;
|
||||||
|
|
||||||
|
import javax.annotation.Nonnull;
|
||||||
|
import java.util.function.Consumer;
|
||||||
|
|
||||||
|
@ExtendWith(MockitoExtension.class)
|
||||||
|
public abstract class BaseBatch2Test {
|
||||||
|
|
||||||
|
public static final String JOB_DEFINITION_ID = "JOB_DEFINITION_ID";
|
||||||
|
public static final String PARAM_1_VALUE = "PARAM 1 VALUE";
|
||||||
|
public static final String PARAM_2_VALUE = "PARAM 2 VALUE";
|
||||||
|
public static final String PASSWORD_VALUE = "PASSWORD VALUE";
|
||||||
|
public static final String STEP_1 = "STEP_1";
|
||||||
|
public static final String STEP_2 = "STEP_2";
|
||||||
|
public static final String STEP_3 = "STEP_3";
|
||||||
|
public static final String INSTANCE_ID = "INSTANCE-ID";
|
||||||
|
public static final String CHUNK_ID = "CHUNK-ID";
|
||||||
|
public static final String CHUNK_ID_2 = "CHUNK-ID-2";
|
||||||
|
public static final String DATA_1_VALUE = "data 1 value";
|
||||||
|
public static final String DATA_2_VALUE = "data 2 value";
|
||||||
|
public static final String DATA_3_VALUE = "data 3 value";
|
||||||
|
public static final String DATA_4_VALUE = "data 4 value";
|
||||||
|
|
||||||
|
@Mock
|
||||||
|
protected IJobStepWorker<TestJobParameters, VoidModel, TestJobStep2InputType> myStep1Worker;
|
||||||
|
@Mock
|
||||||
|
protected IJobStepWorker<TestJobParameters, TestJobStep2InputType, TestJobStep3InputType> myStep2Worker;
|
||||||
|
@Mock
|
||||||
|
protected IJobStepWorker<TestJobParameters, TestJobStep3InputType, VoidModel> myStep3Worker;
|
||||||
|
|
||||||
|
@Nonnull
|
||||||
|
static JobInstance createInstance() {
|
||||||
|
JobInstance instance = new JobInstance();
|
||||||
|
instance.setInstanceId(INSTANCE_ID);
|
||||||
|
instance.setStatus(StatusEnum.IN_PROGRESS);
|
||||||
|
instance.setJobDefinitionId(JOB_DEFINITION_ID);
|
||||||
|
instance.setJobDefinitionVersion(1);
|
||||||
|
instance.setParameters(new TestJobParameters()
|
||||||
|
.setParam1(PARAM_1_VALUE)
|
||||||
|
.setParam2(PARAM_2_VALUE)
|
||||||
|
.setPassword(PASSWORD_VALUE)
|
||||||
|
);
|
||||||
|
return instance;
|
||||||
|
}
|
||||||
|
|
||||||
|
@SafeVarargs
|
||||||
|
final JobDefinition<TestJobParameters> createJobDefinition(Consumer<JobDefinition.Builder<TestJobParameters, ?>>... theModifiers) {
|
||||||
|
JobDefinition.Builder<TestJobParameters, VoidModel> builder = JobDefinition
|
||||||
|
.newBuilder()
|
||||||
|
.setJobDefinitionId(JOB_DEFINITION_ID)
|
||||||
|
.setJobDescription("This is a job description")
|
||||||
|
.setJobDefinitionVersion(1)
|
||||||
|
.setParametersType(TestJobParameters.class)
|
||||||
|
.addFirstStep(STEP_1, "Step 1", TestJobStep2InputType.class, myStep1Worker)
|
||||||
|
.addIntermediateStep(STEP_2, "Step 2", TestJobStep3InputType.class, myStep2Worker)
|
||||||
|
.addLastStep(STEP_3, "Step 3", myStep3Worker);
|
||||||
|
|
||||||
|
for (Consumer<JobDefinition.Builder<TestJobParameters, ?>> next : theModifiers) {
|
||||||
|
next.accept(builder);
|
||||||
|
}
|
||||||
|
|
||||||
|
return builder.build();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -3,7 +3,6 @@ package ca.uhn.fhir.batch2.impl;
|
||||||
import ca.uhn.fhir.batch2.api.IJobDataSink;
|
import ca.uhn.fhir.batch2.api.IJobDataSink;
|
||||||
import ca.uhn.fhir.batch2.api.IJobParametersValidator;
|
import ca.uhn.fhir.batch2.api.IJobParametersValidator;
|
||||||
import ca.uhn.fhir.batch2.api.IJobPersistence;
|
import ca.uhn.fhir.batch2.api.IJobPersistence;
|
||||||
import ca.uhn.fhir.batch2.api.IJobStepWorker;
|
|
||||||
import ca.uhn.fhir.batch2.api.RunOutcome;
|
import ca.uhn.fhir.batch2.api.RunOutcome;
|
||||||
import ca.uhn.fhir.batch2.api.StepExecutionDetails;
|
import ca.uhn.fhir.batch2.api.StepExecutionDetails;
|
||||||
import ca.uhn.fhir.batch2.api.VoidModel;
|
import ca.uhn.fhir.batch2.api.VoidModel;
|
||||||
|
@ -35,12 +34,12 @@ import org.springframework.messaging.MessageDeliveryException;
|
||||||
import javax.annotation.Nonnull;
|
import javax.annotation.Nonnull;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Optional;
|
import java.util.Optional;
|
||||||
import java.util.function.Consumer;
|
|
||||||
|
|
||||||
import static org.junit.jupiter.api.Assertions.assertEquals;
|
import static org.junit.jupiter.api.Assertions.assertEquals;
|
||||||
import static org.junit.jupiter.api.Assertions.assertNull;
|
import static org.junit.jupiter.api.Assertions.assertNull;
|
||||||
import static org.junit.jupiter.api.Assertions.fail;
|
import static org.junit.jupiter.api.Assertions.fail;
|
||||||
import static org.mockito.ArgumentMatchers.any;
|
import static org.mockito.ArgumentMatchers.any;
|
||||||
|
import static org.mockito.ArgumentMatchers.anyInt;
|
||||||
import static org.mockito.ArgumentMatchers.eq;
|
import static org.mockito.ArgumentMatchers.eq;
|
||||||
import static org.mockito.Mockito.times;
|
import static org.mockito.Mockito.times;
|
||||||
import static org.mockito.Mockito.verify;
|
import static org.mockito.Mockito.verify;
|
||||||
|
@ -49,21 +48,8 @@ import static org.mockito.Mockito.when;
|
||||||
|
|
||||||
@ExtendWith(MockitoExtension.class)
|
@ExtendWith(MockitoExtension.class)
|
||||||
@TestMethodOrder(MethodOrderer.MethodName.class)
|
@TestMethodOrder(MethodOrderer.MethodName.class)
|
||||||
public class JobCoordinatorImplTest {
|
public class JobCoordinatorImplTest extends BaseBatch2Test {
|
||||||
|
|
||||||
public static final String JOB_DEFINITION_ID = "JOB_DEFINITION_ID";
|
|
||||||
public static final String PARAM_1_VALUE = "PARAM 1 VALUE";
|
|
||||||
public static final String PARAM_2_VALUE = "PARAM 2 VALUE";
|
|
||||||
public static final String PASSWORD_VALUE = "PASSWORD VALUE";
|
|
||||||
public static final String STEP_1 = "STEP_1";
|
|
||||||
public static final String STEP_2 = "STEP_2";
|
|
||||||
public static final String STEP_3 = "STEP_3";
|
|
||||||
public static final String INSTANCE_ID = "INSTANCE-ID";
|
|
||||||
public static final String CHUNK_ID = "CHUNK-ID";
|
|
||||||
public static final String DATA_1_VALUE = "data 1 value";
|
|
||||||
public static final String DATA_2_VALUE = "data 2 value";
|
|
||||||
public static final String DATA_3_VALUE = "data 3 value";
|
|
||||||
public static final String DATA_4_VALUE = "data 4 value";
|
|
||||||
private static final Logger ourLog = LoggerFactory.getLogger(JobCoordinatorImplTest.class);
|
private static final Logger ourLog = LoggerFactory.getLogger(JobCoordinatorImplTest.class);
|
||||||
private final IChannelReceiver myWorkChannelReceiver = LinkedBlockingChannel.newSynchronous("receiver");
|
private final IChannelReceiver myWorkChannelReceiver = LinkedBlockingChannel.newSynchronous("receiver");
|
||||||
private JobCoordinatorImpl mySvc;
|
private JobCoordinatorImpl mySvc;
|
||||||
|
@ -73,12 +59,6 @@ public class JobCoordinatorImplTest {
|
||||||
private IJobPersistence myJobInstancePersister;
|
private IJobPersistence myJobInstancePersister;
|
||||||
@Mock
|
@Mock
|
||||||
private JobDefinitionRegistry myJobDefinitionRegistry;
|
private JobDefinitionRegistry myJobDefinitionRegistry;
|
||||||
@Mock
|
|
||||||
private IJobStepWorker<TestJobParameters, VoidModel, TestJobStep2InputType> myStep1Worker;
|
|
||||||
@Mock
|
|
||||||
private IJobStepWorker<TestJobParameters, TestJobStep2InputType, TestJobStep3InputType> myStep2Worker;
|
|
||||||
@Mock
|
|
||||||
private IJobStepWorker<TestJobParameters, TestJobStep3InputType, VoidModel> myStep3Worker;
|
|
||||||
@Captor
|
@Captor
|
||||||
private ArgumentCaptor<StepExecutionDetails<TestJobParameters, VoidModel>> myStep1ExecutionDetailsCaptor;
|
private ArgumentCaptor<StepExecutionDetails<TestJobParameters, VoidModel>> myStep1ExecutionDetailsCaptor;
|
||||||
@Captor
|
@Captor
|
||||||
|
@ -114,7 +94,7 @@ public class JobCoordinatorImplTest {
|
||||||
|
|
||||||
// Setup
|
// Setup
|
||||||
|
|
||||||
JobDefinition definition = createJobDefinition();
|
JobDefinition<?> definition = createJobDefinition();
|
||||||
JobInstance instance = createInstance();
|
JobInstance instance = createInstance();
|
||||||
|
|
||||||
when(myJobDefinitionRegistry.getJobDefinition(eq(JOB_DEFINITION_ID), eq(1))).thenReturn(Optional.of(definition));
|
when(myJobDefinitionRegistry.getJobDefinition(eq(JOB_DEFINITION_ID), eq(1))).thenReturn(Optional.of(definition));
|
||||||
|
@ -160,7 +140,8 @@ public class JobCoordinatorImplTest {
|
||||||
when(myJobInstancePersister.fetchWorkChunkSetStartTimeAndMarkInProgress(eq(CHUNK_ID))).thenReturn(Optional.of(createWorkChunkStep1()));
|
when(myJobInstancePersister.fetchWorkChunkSetStartTimeAndMarkInProgress(eq(CHUNK_ID))).thenReturn(Optional.of(createWorkChunkStep1()));
|
||||||
when(myStep1Worker.run(any(), any())).thenAnswer(t -> {
|
when(myStep1Worker.run(any(), any())).thenAnswer(t -> {
|
||||||
IJobDataSink<TestJobStep2InputType> sink = t.getArgument(1, IJobDataSink.class);
|
IJobDataSink<TestJobStep2InputType> sink = t.getArgument(1, IJobDataSink.class);
|
||||||
sink.accept(new TestJobStep2InputType("data value 1", "data value 2"));
|
sink.accept(new TestJobStep2InputType("data value 1a", "data value 2a"));
|
||||||
|
sink.accept(new TestJobStep2InputType("data value 1b", "data value 2b"));
|
||||||
return new RunOutcome(50);
|
return new RunOutcome(50);
|
||||||
});
|
});
|
||||||
mySvc.start();
|
mySvc.start();
|
||||||
|
@ -178,6 +159,8 @@ public class JobCoordinatorImplTest {
|
||||||
assertEquals(PASSWORD_VALUE, params.getPassword());
|
assertEquals(PASSWORD_VALUE, params.getPassword());
|
||||||
|
|
||||||
verify(myJobInstancePersister, times(1)).markWorkChunkAsCompletedAndClearData(any(), eq(50));
|
verify(myJobInstancePersister, times(1)).markWorkChunkAsCompletedAndClearData(any(), eq(50));
|
||||||
|
verify(myJobInstancePersister, times(0)).fetchWorkChunksWithoutData(any(), anyInt(), anyInt());
|
||||||
|
verify(myBatchJobSender, times(2)).sendWorkChannelMessage(any());
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -210,6 +193,41 @@ public class JobCoordinatorImplTest {
|
||||||
verify(myJobInstancePersister, times(1)).markInstanceAsCompleted(eq(INSTANCE_ID));
|
verify(myJobInstancePersister, times(1)).markInstanceAsCompleted(eq(INSTANCE_ID));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testPerformStep_FirstStep_GatedExecutionMode() {
|
||||||
|
|
||||||
|
// Setup
|
||||||
|
|
||||||
|
JobDefinition<TestJobParameters> jobDefinition = createJobDefinition(
|
||||||
|
t -> t.gatedExecution()
|
||||||
|
);
|
||||||
|
when(myJobDefinitionRegistry.getJobDefinition(eq(JOB_DEFINITION_ID), eq(1))).thenReturn(Optional.of(jobDefinition));
|
||||||
|
when(myJobInstancePersister.fetchInstanceAndMarkInProgress(eq(INSTANCE_ID))).thenReturn(Optional.of(createInstance()));
|
||||||
|
when(myJobInstancePersister.fetchWorkChunkSetStartTimeAndMarkInProgress(eq(CHUNK_ID))).thenReturn(Optional.of(createWorkChunkStep1()));
|
||||||
|
when(myStep1Worker.run(any(), any())).thenAnswer(t -> {
|
||||||
|
IJobDataSink<TestJobStep2InputType> sink = t.getArgument(1, IJobDataSink.class);
|
||||||
|
sink.accept(new TestJobStep2InputType("data value 1a", "data value 2a"));
|
||||||
|
sink.accept(new TestJobStep2InputType("data value 1b", "data value 2b"));
|
||||||
|
return new RunOutcome(50);
|
||||||
|
});
|
||||||
|
mySvc.start();
|
||||||
|
|
||||||
|
// Execute
|
||||||
|
|
||||||
|
myWorkChannelReceiver.send(new JobWorkNotificationJsonMessage(createWorkNotification(STEP_1)));
|
||||||
|
|
||||||
|
// Verify
|
||||||
|
|
||||||
|
verify(myStep1Worker, times(1)).run(myStep1ExecutionDetailsCaptor.capture(), any());
|
||||||
|
TestJobParameters params = myStep1ExecutionDetailsCaptor.getValue().getParameters();
|
||||||
|
assertEquals(PARAM_1_VALUE, params.getParam1());
|
||||||
|
assertEquals(PARAM_2_VALUE, params.getParam2());
|
||||||
|
assertEquals(PASSWORD_VALUE, params.getPassword());
|
||||||
|
|
||||||
|
verify(myJobInstancePersister, times(1)).markWorkChunkAsCompletedAndClearData(any(), eq(50));
|
||||||
|
verify(myBatchJobSender, times(0)).sendWorkChannelMessage(any());
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testPerformStep_SecondStep() {
|
public void testPerformStep_SecondStep() {
|
||||||
|
|
||||||
|
@ -276,7 +294,7 @@ public class JobCoordinatorImplTest {
|
||||||
when(myJobInstancePersister.fetchWorkChunkSetStartTimeAndMarkInProgress(eq(CHUNK_ID))).thenReturn(Optional.of(createWorkChunk(STEP_2, new TestJobStep2InputType(DATA_1_VALUE, DATA_2_VALUE))));
|
when(myJobInstancePersister.fetchWorkChunkSetStartTimeAndMarkInProgress(eq(CHUNK_ID))).thenReturn(Optional.of(createWorkChunk(STEP_2, new TestJobStep2InputType(DATA_1_VALUE, DATA_2_VALUE))));
|
||||||
when(myJobDefinitionRegistry.getJobDefinition(eq(JOB_DEFINITION_ID), eq(1))).thenReturn(Optional.of(createJobDefinition()));
|
when(myJobDefinitionRegistry.getJobDefinition(eq(JOB_DEFINITION_ID), eq(1))).thenReturn(Optional.of(createJobDefinition()));
|
||||||
when(myJobInstancePersister.fetchInstanceAndMarkInProgress(eq(INSTANCE_ID))).thenReturn(Optional.of(createInstance()));
|
when(myJobInstancePersister.fetchInstanceAndMarkInProgress(eq(INSTANCE_ID))).thenReturn(Optional.of(createInstance()));
|
||||||
when(myStep2Worker.run(any(), any())).thenAnswer(t->{
|
when(myStep2Worker.run(any(), any())).thenAnswer(t -> {
|
||||||
IJobDataSink<?> sink = t.getArgument(1, IJobDataSink.class);
|
IJobDataSink<?> sink = t.getArgument(1, IJobDataSink.class);
|
||||||
sink.recoveredError("Error message 1");
|
sink.recoveredError("Error message 1");
|
||||||
sink.recoveredError("Error message 2");
|
sink.recoveredError("Error message 2");
|
||||||
|
@ -479,7 +497,7 @@ public class JobCoordinatorImplTest {
|
||||||
}
|
}
|
||||||
return null;
|
return null;
|
||||||
};
|
};
|
||||||
JobDefinition<?> jobDefinition = createJobDefinition(t->t.setParametersValidator(v));
|
JobDefinition<?> jobDefinition = createJobDefinition(t -> t.setParametersValidator(v));
|
||||||
when(myJobDefinitionRegistry.getLatestJobDefinition(eq(JOB_DEFINITION_ID))).thenReturn(Optional.of(jobDefinition));
|
when(myJobDefinitionRegistry.getLatestJobDefinition(eq(JOB_DEFINITION_ID))).thenReturn(Optional.of(jobDefinition));
|
||||||
|
|
||||||
// Execute
|
// Execute
|
||||||
|
@ -504,51 +522,17 @@ public class JobCoordinatorImplTest {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@SafeVarargs
|
|
||||||
private JobDefinition<TestJobParameters> createJobDefinition(Consumer<JobDefinition.Builder<TestJobParameters, ?>>... theModifiers) {
|
|
||||||
JobDefinition.Builder<TestJobParameters, VoidModel> builder = JobDefinition
|
|
||||||
.newBuilder()
|
|
||||||
.setJobDefinitionId(JOB_DEFINITION_ID)
|
|
||||||
.setJobDescription("This is a job description")
|
|
||||||
.setJobDefinitionVersion(1)
|
|
||||||
.setParametersType(TestJobParameters.class)
|
|
||||||
.addFirstStep(STEP_1, "Step 1", TestJobStep2InputType.class, myStep1Worker)
|
|
||||||
.addIntermediateStep(STEP_2, "Step 2", TestJobStep3InputType.class, myStep2Worker)
|
|
||||||
.addLastStep(STEP_3, "Step 3", myStep3Worker);
|
|
||||||
|
|
||||||
for (Consumer<JobDefinition.Builder<TestJobParameters, ?>> next : theModifiers) {
|
|
||||||
next.accept(builder);
|
|
||||||
}
|
|
||||||
|
|
||||||
return builder.build();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Nonnull
|
@Nonnull
|
||||||
private JobWorkNotification createWorkNotification(String theStepId) {
|
private JobWorkNotification createWorkNotification(String theStepId) {
|
||||||
JobWorkNotification payload = new JobWorkNotification();
|
JobWorkNotification payload = new JobWorkNotification();
|
||||||
payload.setJobDefinitionId(JOB_DEFINITION_ID);
|
payload.setJobDefinitionId(JOB_DEFINITION_ID);
|
||||||
payload.setJobDefinitionVersion(1);
|
payload.setJobDefinitionVersion(1);
|
||||||
payload.setInstanceId(INSTANCE_ID);
|
payload.setInstanceId(INSTANCE_ID);
|
||||||
payload.setChunkId(JobCoordinatorImplTest.CHUNK_ID);
|
payload.setChunkId(BaseBatch2Test.CHUNK_ID);
|
||||||
payload.setTargetStepId(theStepId);
|
payload.setTargetStepId(theStepId);
|
||||||
return payload;
|
return payload;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Nonnull
|
|
||||||
static JobInstance createInstance() {
|
|
||||||
JobInstance instance = new JobInstance();
|
|
||||||
instance.setInstanceId(INSTANCE_ID);
|
|
||||||
instance.setStatus(StatusEnum.IN_PROGRESS);
|
|
||||||
instance.setJobDefinitionId(JOB_DEFINITION_ID);
|
|
||||||
instance.setJobDefinitionVersion(1);
|
|
||||||
instance.setParameters(new TestJobParameters()
|
|
||||||
.setParam1(PARAM_1_VALUE)
|
|
||||||
.setParam2(PARAM_2_VALUE)
|
|
||||||
.setPassword(PASSWORD_VALUE)
|
|
||||||
);
|
|
||||||
return instance;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Nonnull
|
@Nonnull
|
||||||
static WorkChunk createWorkChunk(String theTargetStepId, IModelJson theData) {
|
static WorkChunk createWorkChunk(String theTargetStepId, IModelJson theData) {
|
||||||
return new WorkChunk()
|
return new WorkChunk()
|
||||||
|
|
|
@ -88,7 +88,7 @@ class JobDataSinkTest {
|
||||||
// Let's test our first step worker by calling run on it:
|
// Let's test our first step worker by calling run on it:
|
||||||
when(myJobPersistence.storeWorkChunk(myBatchWorkChunkCaptor.capture())).thenReturn(CHUNK_ID);
|
when(myJobPersistence.storeWorkChunk(myBatchWorkChunkCaptor.capture())).thenReturn(CHUNK_ID);
|
||||||
StepExecutionDetails<TestJobParameters, VoidModel> details = new StepExecutionDetails<>(new TestJobParameters().setParam1("" + PID_COUNT), null, JOB_INSTANCE_ID, CHUNK_ID);
|
StepExecutionDetails<TestJobParameters, VoidModel> details = new StepExecutionDetails<>(new TestJobParameters().setParam1("" + PID_COUNT), null, JOB_INSTANCE_ID, CHUNK_ID);
|
||||||
JobDataSink<Step1Output> sink = new JobDataSink<>(myBatchJobSender, myJobPersistence, JOB_DEF_ID, JOB_DEF_VERSION, job.getSteps().get(1), JOB_INSTANCE_ID, FIRST_STEP_ID);
|
JobDataSink<Step1Output> sink = new JobDataSink<>(myBatchJobSender, myJobPersistence, JOB_DEF_ID, JOB_DEF_VERSION, job.getSteps().get(1), JOB_INSTANCE_ID, FIRST_STEP_ID, false);
|
||||||
|
|
||||||
RunOutcome result = firstStepWorker.run(details, sink);
|
RunOutcome result = firstStepWorker.run(details, sink);
|
||||||
|
|
||||||
|
|
|
@ -1,9 +1,13 @@
|
||||||
package ca.uhn.fhir.batch2.impl;
|
package ca.uhn.fhir.batch2.impl;
|
||||||
|
|
||||||
|
import ca.uhn.fhir.batch2.api.IJobCompletionHandler;
|
||||||
import ca.uhn.fhir.batch2.api.IJobPersistence;
|
import ca.uhn.fhir.batch2.api.IJobPersistence;
|
||||||
|
import ca.uhn.fhir.batch2.api.JobCompletionDetails;
|
||||||
import ca.uhn.fhir.batch2.model.JobInstance;
|
import ca.uhn.fhir.batch2.model.JobInstance;
|
||||||
|
import ca.uhn.fhir.batch2.model.JobWorkNotification;
|
||||||
import ca.uhn.fhir.batch2.model.StatusEnum;
|
import ca.uhn.fhir.batch2.model.StatusEnum;
|
||||||
import ca.uhn.fhir.jpa.model.sched.ISchedulerService;
|
import ca.uhn.fhir.jpa.model.sched.ISchedulerService;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.channel.api.IChannelProducer;
|
||||||
import com.google.common.collect.Lists;
|
import com.google.common.collect.Lists;
|
||||||
import org.hl7.fhir.r4.model.DateTimeType;
|
import org.hl7.fhir.r4.model.DateTimeType;
|
||||||
import org.junit.jupiter.api.BeforeEach;
|
import org.junit.jupiter.api.BeforeEach;
|
||||||
|
@ -13,14 +17,10 @@ import org.mockito.ArgumentCaptor;
|
||||||
import org.mockito.Captor;
|
import org.mockito.Captor;
|
||||||
import org.mockito.Mock;
|
import org.mockito.Mock;
|
||||||
import org.mockito.junit.jupiter.MockitoExtension;
|
import org.mockito.junit.jupiter.MockitoExtension;
|
||||||
|
import org.springframework.messaging.Message;
|
||||||
|
|
||||||
import java.util.Date;
|
import java.util.Date;
|
||||||
|
|
||||||
import static ca.uhn.fhir.batch2.impl.JobCoordinatorImplTest.INSTANCE_ID;
|
|
||||||
import static ca.uhn.fhir.batch2.impl.JobCoordinatorImplTest.STEP_1;
|
|
||||||
import static ca.uhn.fhir.batch2.impl.JobCoordinatorImplTest.STEP_2;
|
|
||||||
import static ca.uhn.fhir.batch2.impl.JobCoordinatorImplTest.STEP_3;
|
|
||||||
import static ca.uhn.fhir.batch2.impl.JobCoordinatorImplTest.createInstance;
|
|
||||||
import static ca.uhn.fhir.batch2.impl.JobCoordinatorImplTest.createWorkChunk;
|
import static ca.uhn.fhir.batch2.impl.JobCoordinatorImplTest.createWorkChunk;
|
||||||
import static ca.uhn.fhir.batch2.impl.JobCoordinatorImplTest.createWorkChunkStep1;
|
import static ca.uhn.fhir.batch2.impl.JobCoordinatorImplTest.createWorkChunkStep1;
|
||||||
import static ca.uhn.fhir.batch2.impl.JobCoordinatorImplTest.createWorkChunkStep2;
|
import static ca.uhn.fhir.batch2.impl.JobCoordinatorImplTest.createWorkChunkStep2;
|
||||||
|
@ -38,35 +38,48 @@ import static org.mockito.Mockito.verifyNoMoreInteractions;
|
||||||
import static org.mockito.Mockito.when;
|
import static org.mockito.Mockito.when;
|
||||||
|
|
||||||
@ExtendWith(MockitoExtension.class)
|
@ExtendWith(MockitoExtension.class)
|
||||||
public class JobCleanerServiceImplTest {
|
public class JobMaintenanceServiceImplTest extends BaseBatch2Test {
|
||||||
|
|
||||||
|
@Mock
|
||||||
|
IJobCompletionHandler<TestJobParameters> myCompletionHandler;
|
||||||
@Mock
|
@Mock
|
||||||
private ISchedulerService mySchedulerService;
|
private ISchedulerService mySchedulerService;
|
||||||
@Mock
|
@Mock
|
||||||
private IJobPersistence myJobPersistence;
|
private IJobPersistence myJobPersistence;
|
||||||
private JobCleanerServiceImpl mySvc;
|
private JobMaintenanceServiceImpl mySvc;
|
||||||
@Captor
|
@Captor
|
||||||
private ArgumentCaptor<JobInstance> myInstanceCaptor;
|
private ArgumentCaptor<JobInstance> myInstanceCaptor;
|
||||||
|
private JobDefinitionRegistry myJobDefinitionRegistry;
|
||||||
|
@Mock
|
||||||
|
private IChannelProducer myWorkChannelProducer;
|
||||||
|
@Captor
|
||||||
|
private ArgumentCaptor<Message<JobWorkNotification>> myMessageCaptor;
|
||||||
|
@Captor
|
||||||
|
private ArgumentCaptor<JobCompletionDetails<TestJobParameters>> myJobCompletionCaptor;
|
||||||
|
|
||||||
@BeforeEach
|
@BeforeEach
|
||||||
public void beforeEach() {
|
public void beforeEach() {
|
||||||
mySvc = new JobCleanerServiceImpl(mySchedulerService, myJobPersistence);
|
myJobDefinitionRegistry = new JobDefinitionRegistry();
|
||||||
|
BatchJobSender batchJobSender = new BatchJobSender(myWorkChannelProducer);
|
||||||
|
mySvc = new JobMaintenanceServiceImpl(mySchedulerService, myJobPersistence, myJobDefinitionRegistry, batchJobSender);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testInProgress_CalculateProgress_FirstCompleteButNoOtherStepsYetComplete() {
|
public void testInProgress_CalculateProgress_FirstCompleteButNoOtherStepsYetComplete() {
|
||||||
|
myJobDefinitionRegistry.addJobDefinition(createJobDefinition());
|
||||||
when(myJobPersistence.fetchInstances(anyInt(), eq(0))).thenReturn(Lists.newArrayList(createInstance()));
|
when(myJobPersistence.fetchInstances(anyInt(), eq(0))).thenReturn(Lists.newArrayList(createInstance()));
|
||||||
when(myJobPersistence.fetchWorkChunksWithoutData(eq(INSTANCE_ID), anyInt(), eq(0))).thenReturn(Lists.newArrayList(
|
when(myJobPersistence.fetchWorkChunksWithoutData(eq(INSTANCE_ID), anyInt(), eq(0))).thenReturn(Lists.newArrayList(
|
||||||
createWorkChunk(STEP_1, null).setStatus(StatusEnum.COMPLETED)
|
createWorkChunk(STEP_1, null).setStatus(StatusEnum.COMPLETED)
|
||||||
));
|
));
|
||||||
|
|
||||||
mySvc.runCleanupPass();
|
mySvc.runMaintenancePass();
|
||||||
|
|
||||||
verify(myJobPersistence, never()).updateInstance(any());
|
verify(myJobPersistence, never()).updateInstance(any());
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testInProgress_CalculateProgress_FirstStepComplete() {
|
public void testInProgress_CalculateProgress_FirstStepComplete() {
|
||||||
|
myJobDefinitionRegistry.addJobDefinition(createJobDefinition());
|
||||||
when(myJobPersistence.fetchInstances(anyInt(), eq(0))).thenReturn(Lists.newArrayList(createInstance()));
|
when(myJobPersistence.fetchInstances(anyInt(), eq(0))).thenReturn(Lists.newArrayList(createInstance()));
|
||||||
when(myJobPersistence.fetchWorkChunksWithoutData(eq(INSTANCE_ID), anyInt(), eq(0))).thenReturn(Lists.newArrayList(
|
when(myJobPersistence.fetchWorkChunksWithoutData(eq(INSTANCE_ID), anyInt(), eq(0))).thenReturn(Lists.newArrayList(
|
||||||
createWorkChunkStep1().setStatus(StatusEnum.COMPLETED).setStartTime(parseTime("2022-02-12T14:00:00-04:00")),
|
createWorkChunkStep1().setStatus(StatusEnum.COMPLETED).setStartTime(parseTime("2022-02-12T14:00:00-04:00")),
|
||||||
|
@ -77,7 +90,7 @@ public class JobCleanerServiceImplTest {
|
||||||
createWorkChunkStep3().setStatus(StatusEnum.COMPLETED).setStartTime(parseTime("2022-02-12T14:01:00-04:00")).setEndTime(parseTime("2022-02-12T14:10:00-04:00")).setRecordsProcessed(25)
|
createWorkChunkStep3().setStatus(StatusEnum.COMPLETED).setStartTime(parseTime("2022-02-12T14:01:00-04:00")).setEndTime(parseTime("2022-02-12T14:10:00-04:00")).setRecordsProcessed(25)
|
||||||
));
|
));
|
||||||
|
|
||||||
mySvc.runCleanupPass();
|
mySvc.runMaintenancePass();
|
||||||
|
|
||||||
verify(myJobPersistence, times(1)).updateInstance(myInstanceCaptor.capture());
|
verify(myJobPersistence, times(1)).updateInstance(myInstanceCaptor.capture());
|
||||||
JobInstance instance = myInstanceCaptor.getValue();
|
JobInstance instance = myInstanceCaptor.getValue();
|
||||||
|
@ -87,7 +100,7 @@ public class JobCleanerServiceImplTest {
|
||||||
assertEquals(0.08333333333333333, instance.getCombinedRecordsProcessedPerSecond());
|
assertEquals(0.08333333333333333, instance.getCombinedRecordsProcessedPerSecond());
|
||||||
assertNotNull(instance.getStartTime());
|
assertNotNull(instance.getStartTime());
|
||||||
assertEquals(parseTime("2022-02-12T14:00:00-04:00"), instance.getStartTime());
|
assertEquals(parseTime("2022-02-12T14:00:00-04:00"), instance.getStartTime());
|
||||||
assertEquals(null, instance.getEndTime());
|
assertNull(instance.getEndTime());
|
||||||
assertEquals("00:10:00", instance.getEstimatedTimeRemaining());
|
assertEquals("00:10:00", instance.getEstimatedTimeRemaining());
|
||||||
|
|
||||||
verifyNoMoreInteractions(myJobPersistence);
|
verifyNoMoreInteractions(myJobPersistence);
|
||||||
|
@ -95,6 +108,8 @@ public class JobCleanerServiceImplTest {
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testInProgress_CalculateProgress_InstanceHasErrorButNoChunksAreErrored() {
|
public void testInProgress_CalculateProgress_InstanceHasErrorButNoChunksAreErrored() {
|
||||||
|
// Setup
|
||||||
|
myJobDefinitionRegistry.addJobDefinition(createJobDefinition());
|
||||||
JobInstance instance1 = createInstance();
|
JobInstance instance1 = createInstance();
|
||||||
instance1.setErrorMessage("This is an error message");
|
instance1.setErrorMessage("This is an error message");
|
||||||
when(myJobPersistence.fetchInstances(anyInt(), eq(0))).thenReturn(Lists.newArrayList(instance1));
|
when(myJobPersistence.fetchInstances(anyInt(), eq(0))).thenReturn(Lists.newArrayList(instance1));
|
||||||
|
@ -107,8 +122,10 @@ public class JobCleanerServiceImplTest {
|
||||||
createWorkChunkStep3().setStatus(StatusEnum.COMPLETED).setStartTime(parseTime("2022-02-12T14:01:00-04:00")).setEndTime(parseTime("2022-02-12T14:10:00-04:00")).setRecordsProcessed(25)
|
createWorkChunkStep3().setStatus(StatusEnum.COMPLETED).setStartTime(parseTime("2022-02-12T14:01:00-04:00")).setEndTime(parseTime("2022-02-12T14:10:00-04:00")).setRecordsProcessed(25)
|
||||||
));
|
));
|
||||||
|
|
||||||
mySvc.runCleanupPass();
|
// Execute
|
||||||
|
mySvc.runMaintenancePass();
|
||||||
|
|
||||||
|
// Verify
|
||||||
verify(myJobPersistence, times(1)).updateInstance(myInstanceCaptor.capture());
|
verify(myJobPersistence, times(1)).updateInstance(myInstanceCaptor.capture());
|
||||||
JobInstance instance = myInstanceCaptor.getValue();
|
JobInstance instance = myInstanceCaptor.getValue();
|
||||||
|
|
||||||
|
@ -121,6 +138,30 @@ public class JobCleanerServiceImplTest {
|
||||||
verifyNoMoreInteractions(myJobPersistence);
|
verifyNoMoreInteractions(myJobPersistence);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testInProgress_GatedExecution_FirstStepComplete() {
|
||||||
|
// Setup
|
||||||
|
myJobDefinitionRegistry.addJobDefinition(createJobDefinition(t -> t.gatedExecution()));
|
||||||
|
when(myJobPersistence.fetchWorkChunksWithoutData(eq(INSTANCE_ID), eq(100), eq(0))).thenReturn(Lists.newArrayList(
|
||||||
|
createWorkChunkStep2().setStatus(StatusEnum.QUEUED).setId(CHUNK_ID),
|
||||||
|
createWorkChunkStep2().setStatus(StatusEnum.QUEUED).setId(CHUNK_ID_2)
|
||||||
|
));
|
||||||
|
JobInstance instance1 = createInstance();
|
||||||
|
instance1.setCurrentGatedStepId(STEP_1);
|
||||||
|
when(myJobPersistence.fetchInstances(anyInt(), eq(0))).thenReturn(Lists.newArrayList(instance1));
|
||||||
|
|
||||||
|
// Execute
|
||||||
|
mySvc.runMaintenancePass();
|
||||||
|
|
||||||
|
// Verify
|
||||||
|
verify(myWorkChannelProducer, times(2)).send(myMessageCaptor.capture());
|
||||||
|
JobWorkNotification payload0 = myMessageCaptor.getAllValues().get(0).getPayload();
|
||||||
|
assertEquals(STEP_2, payload0.getTargetStepId());
|
||||||
|
assertEquals(CHUNK_ID, payload0.getChunkId());
|
||||||
|
JobWorkNotification payload1 = myMessageCaptor.getAllValues().get(1).getPayload();
|
||||||
|
assertEquals(STEP_2, payload1.getTargetStepId());
|
||||||
|
assertEquals(CHUNK_ID_2, payload1.getChunkId());
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testFailed_PurgeOldInstance() {
|
public void testFailed_PurgeOldInstance() {
|
||||||
|
@ -129,7 +170,7 @@ public class JobCleanerServiceImplTest {
|
||||||
instance.setEndTime(parseTime("2001-01-01T12:12:12Z"));
|
instance.setEndTime(parseTime("2001-01-01T12:12:12Z"));
|
||||||
when(myJobPersistence.fetchInstances(anyInt(), eq(0))).thenReturn(Lists.newArrayList(instance));
|
when(myJobPersistence.fetchInstances(anyInt(), eq(0))).thenReturn(Lists.newArrayList(instance));
|
||||||
|
|
||||||
mySvc.runCleanupPass();
|
mySvc.runMaintenancePass();
|
||||||
|
|
||||||
verify(myJobPersistence, times(1)).deleteInstanceAndChunks(eq(INSTANCE_ID));
|
verify(myJobPersistence, times(1)).deleteInstanceAndChunks(eq(INSTANCE_ID));
|
||||||
verifyNoMoreInteractions(myJobPersistence);
|
verifyNoMoreInteractions(myJobPersistence);
|
||||||
|
@ -137,6 +178,9 @@ public class JobCleanerServiceImplTest {
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testInProgress_CalculateProgress_AllStepsComplete() {
|
public void testInProgress_CalculateProgress_AllStepsComplete() {
|
||||||
|
// Setup
|
||||||
|
|
||||||
|
myJobDefinitionRegistry.addJobDefinition(createJobDefinition(t -> t.completionHandler(myCompletionHandler)));
|
||||||
when(myJobPersistence.fetchInstances(anyInt(), eq(0))).thenReturn(Lists.newArrayList(createInstance()));
|
when(myJobPersistence.fetchInstances(anyInt(), eq(0))).thenReturn(Lists.newArrayList(createInstance()));
|
||||||
when(myJobPersistence.fetchWorkChunksWithoutData(eq(INSTANCE_ID), anyInt(), eq(0))).thenReturn(Lists.newArrayList(
|
when(myJobPersistence.fetchWorkChunksWithoutData(eq(INSTANCE_ID), anyInt(), eq(0))).thenReturn(Lists.newArrayList(
|
||||||
createWorkChunkStep1().setStatus(StatusEnum.COMPLETED).setStartTime(parseTime("2022-02-12T14:00:00-04:00")).setEndTime(parseTime("2022-02-12T14:01:00-04:00")).setRecordsProcessed(25),
|
createWorkChunkStep1().setStatus(StatusEnum.COMPLETED).setStartTime(parseTime("2022-02-12T14:00:00-04:00")).setEndTime(parseTime("2022-02-12T14:01:00-04:00")).setRecordsProcessed(25),
|
||||||
|
@ -147,7 +191,11 @@ public class JobCleanerServiceImplTest {
|
||||||
createWorkChunkStep3().setStatus(StatusEnum.COMPLETED).setStartTime(parseTime("2022-02-12T14:01:00-04:00")).setEndTime(parseTime("2022-02-12T14:10:00-04:00")).setRecordsProcessed(25)
|
createWorkChunkStep3().setStatus(StatusEnum.COMPLETED).setStartTime(parseTime("2022-02-12T14:01:00-04:00")).setEndTime(parseTime("2022-02-12T14:10:00-04:00")).setRecordsProcessed(25)
|
||||||
));
|
));
|
||||||
|
|
||||||
mySvc.runCleanupPass();
|
// Execute
|
||||||
|
|
||||||
|
mySvc.runMaintenancePass();
|
||||||
|
|
||||||
|
// Verify
|
||||||
|
|
||||||
verify(myJobPersistence, times(2)).updateInstance(myInstanceCaptor.capture());
|
verify(myJobPersistence, times(2)).updateInstance(myInstanceCaptor.capture());
|
||||||
JobInstance instance = myInstanceCaptor.getAllValues().get(0);
|
JobInstance instance = myInstanceCaptor.getAllValues().get(0);
|
||||||
|
@ -159,8 +207,12 @@ public class JobCleanerServiceImplTest {
|
||||||
assertEquals(parseTime("2022-02-12T14:10:00-04:00"), instance.getEndTime());
|
assertEquals(parseTime("2022-02-12T14:10:00-04:00"), instance.getEndTime());
|
||||||
|
|
||||||
verify(myJobPersistence, times(1)).deleteChunks(eq(INSTANCE_ID));
|
verify(myJobPersistence, times(1)).deleteChunks(eq(INSTANCE_ID));
|
||||||
|
verify(myCompletionHandler, times(1)).jobComplete(myJobCompletionCaptor.capture());
|
||||||
|
|
||||||
verifyNoMoreInteractions(myJobPersistence);
|
verifyNoMoreInteractions(myJobPersistence);
|
||||||
|
|
||||||
|
assertEquals(INSTANCE_ID, myJobCompletionCaptor.getValue().getInstanceId());
|
||||||
|
assertEquals(PARAM_1_VALUE, myJobCompletionCaptor.getValue().getParameters().getParam1());
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
|
@ -175,7 +227,7 @@ public class JobCleanerServiceImplTest {
|
||||||
createWorkChunkStep3().setStatus(StatusEnum.COMPLETED).setStartTime(parseTime("2022-02-12T14:01:00-04:00")).setEndTime(parseTime("2022-02-12T14:10:00-04:00")).setRecordsProcessed(25)
|
createWorkChunkStep3().setStatus(StatusEnum.COMPLETED).setStartTime(parseTime("2022-02-12T14:01:00-04:00")).setEndTime(parseTime("2022-02-12T14:10:00-04:00")).setRecordsProcessed(25)
|
||||||
));
|
));
|
||||||
|
|
||||||
mySvc.runCleanupPass();
|
mySvc.runMaintenancePass();
|
||||||
|
|
||||||
verify(myJobPersistence, times(2)).updateInstance(myInstanceCaptor.capture());
|
verify(myJobPersistence, times(2)).updateInstance(myInstanceCaptor.capture());
|
||||||
JobInstance instance = myInstanceCaptor.getAllValues().get(0);
|
JobInstance instance = myInstanceCaptor.getAllValues().get(0);
|
Loading…
Reference in New Issue