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

io.datarouter.tasktracker.storage.LongRunningTask Maven / Gradle / Ivy

There is a newer version: 0.0.126
Show newest version
/*
 * Copyright © 2009 HotPads ([email protected])
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package io.datarouter.tasktracker.storage;

import java.time.Duration;
import java.time.ZoneId;
import java.util.List;
import java.util.Optional;
import java.util.function.Supplier;

import io.datarouter.instrumentation.task.TaskTrackerDto;
import io.datarouter.model.databean.BaseDatabean;
import io.datarouter.model.field.Field;
import io.datarouter.model.field.codec.MilliTimeFieldCodec;
import io.datarouter.model.field.codec.StringMappedEnumFieldCodec;
import io.datarouter.model.field.imp.StringEncodedField;
import io.datarouter.model.field.imp.StringEncodedFieldKey;
import io.datarouter.model.field.imp.StringField;
import io.datarouter.model.field.imp.StringFieldKey;
import io.datarouter.model.field.imp.comparable.BooleanFieldKey;
import io.datarouter.model.field.imp.comparable.LongEncodedField;
import io.datarouter.model.field.imp.comparable.LongEncodedFieldKey;
import io.datarouter.model.field.imp.comparable.LongField;
import io.datarouter.model.field.imp.comparable.LongFieldKey;
import io.datarouter.model.serialize.fielder.BaseDatabeanFielder;
import io.datarouter.model.util.CommonFieldSizes;
import io.datarouter.tasktracker.scheduler.LongRunningTaskStatus;
import io.datarouter.tasktracker.scheduler.LongRunningTaskType;
import io.datarouter.tasktracker.service.LongRunningTaskInfo;
import io.datarouter.types.MilliTime;
import io.datarouter.util.ComparableTool;
import io.datarouter.util.time.DurationTool;

public class LongRunningTask extends BaseDatabean{

	public static final Duration HEARTBEAT_STATUS_WARNING = Duration.ofSeconds(2);
	public static final Duration HEARTBEAT_STATUS_STALLED = Duration.ofSeconds(10);

	private LongRunningTaskType type;
	private MilliTime start;
	private MilliTime finish;
	private MilliTime heartbeat;
	private LongRunningTaskStatus jobExecutionStatus;
	private String triggeredBy;
	private Long numItemsProcessed;
	private String lastItemProcessed;
	private String exceptionRecordId;

	public static class FieldKeys{
		public static final StringEncodedFieldKey type = new StringEncodedFieldKey<>(
				"type",
				new StringMappedEnumFieldCodec<>(LongRunningTaskType.BY_PERSISTENT_STRING));
		public static final LongEncodedFieldKey start = new LongEncodedFieldKey<>(
				"start",
				new MilliTimeFieldCodec());
		public static final BooleanFieldKey interrupt = new BooleanFieldKey("interrupt");
		public static final LongEncodedFieldKey finish = new LongEncodedFieldKey<>(
				"finish",
				new MilliTimeFieldCodec());
		public static final LongEncodedFieldKey heartbeat = new LongEncodedFieldKey<>(
				"heartbeat",
				new MilliTimeFieldCodec());
		public static final StringEncodedFieldKey longRunningTaskStatus
				= new StringEncodedFieldKey<>(
				"jobExecutionStatus",
				new StringMappedEnumFieldCodec<>(LongRunningTaskStatus.BY_PERSISTENT_STRING));
		public static final StringFieldKey triggeredBy = new StringFieldKey("triggeredBy");
		public static final LongFieldKey numItemsProcessed = new LongFieldKey("numItemsProcessed");
		public static final StringFieldKey lastItemProcessed = new StringFieldKey("lastItemProcessed")
				.withSize(CommonFieldSizes.INT_LENGTH_LONGTEXT);
		public static final StringFieldKey exceptionRecordId = new StringFieldKey("exceptionRecordId");
	}

	public static class LongRunningTaskFielder extends BaseDatabeanFielder{

		public LongRunningTaskFielder(){
			super(LongRunningTaskKey::new);
		}

		@Override
		public List> getNonKeyFields(LongRunningTask databean){
			return List.of(
					new StringEncodedField<>(FieldKeys.type, databean.type),
					new LongEncodedField<>(FieldKeys.start, databean.start),
					new LongEncodedField<>(FieldKeys.finish, databean.finish),
					new LongEncodedField<>(FieldKeys.heartbeat, databean.heartbeat),
					new StringEncodedField<>(FieldKeys.longRunningTaskStatus, databean.jobExecutionStatus),
					new StringField(FieldKeys.triggeredBy, databean.triggeredBy),
					new LongField(FieldKeys.numItemsProcessed, databean.numItemsProcessed),
					new StringField(FieldKeys.lastItemProcessed, databean.lastItemProcessed),
					new StringField(FieldKeys.exceptionRecordId, databean.exceptionRecordId));
		}
	}

	public LongRunningTask(){
		super(new LongRunningTaskKey());
	}

	public LongRunningTask(LongRunningTaskInfo task){
		super(new LongRunningTaskKey(task.name, MilliTime.ofEpochMilli(task.triggerTimeMs), task.serverName));
		this.type = task.type;
		this.start = Optional.ofNullable(task.startTimeMs)
				.map(MilliTime::ofEpochMilli)
				.orElse(null);
		this.finish = Optional.ofNullable(task.finishTimeMs)
				.map(MilliTime::ofEpochMilli)
				.orElse(null);
		this.heartbeat = Optional.ofNullable(task.heartbeatTimeMs)
				.map(MilliTime::ofEpochMilli)
				.orElse(null);
		this.jobExecutionStatus = task.longRunningTaskStatus;
		this.triggeredBy = task.triggeredBy;
		this.numItemsProcessed = task.numItemsProcessed.get();
		this.lastItemProcessed = task.lastItemProcessed;
		this.exceptionRecordId = task.exceptionRecordId;
	}

	@Override
	public Supplier getKeySupplier(){
		return LongRunningTaskKey::new;
	}

	public Duration getDuration(){
		MilliTime from = start != null
				? start
				: getKey().getTriggerTime();
		MilliTime to;
		if(finish != null){
			to = finish;
		}else if(jobExecutionStatus == LongRunningTaskStatus.RUNNING){
			to = MilliTime.now();
		}else if(heartbeat != null){
			to = heartbeat;
		}else{
			return null;
		}
		long differenceMs = to.minus(from).toEpochMilli();
		return Duration.ofMillis(differenceMs);
	}

	public String getDurationString(){
		Duration duration = getDuration();
		if(duration == null){
			return "Unknown";
		}
		return DurationTool.toString(duration);
	}

	public String getLastHeartbeatString(ZoneId zoneId){
		if(heartbeat == null){
			return "";
		}
		return heartbeat.format(zoneId);
	}

	public String getFinishTimeString(ZoneId zoneId){
		if(finish == null){
			return "";
		}
		return finish.format(zoneId);
	}

	public boolean isRunning(){
		return jobExecutionStatus == LongRunningTaskStatus.RUNNING;
	}

	public boolean isSuccess(){
		return jobExecutionStatus == LongRunningTaskStatus.SUCCESS;
	}

	public boolean isBadState(){
		return jobExecutionStatus.isBadState;
	}

	public LongRunningTaskHeartBeatStatus getHeartbeatStatus(){
		if(heartbeat == null || !isRunning()){
			return null;
		}
		Duration elapsed = DurationTool.sinceDate(heartbeat.toDate());
		if(ComparableTool.gt(elapsed, HEARTBEAT_STATUS_STALLED)){
			return LongRunningTaskHeartBeatStatus.STALLED;
		}else if(ComparableTool.gt(elapsed, HEARTBEAT_STATUS_WARNING)){
			return LongRunningTaskHeartBeatStatus.WARNING;
		}else{
			return LongRunningTaskHeartBeatStatus.OK;
		}
	}

	public MilliTime getStart(){
		return start;
	}

	public Optional findStart(){
		return Optional.ofNullable(start);
	}

	public void setStartTime(MilliTime start){
		this.start = start;
	}

	public MilliTime getFinish(){
		return finish;
	}

	public void setFinishTime(MilliTime finish){
		this.finish = finish;
	}

	public MilliTime getHeartbeat(){
		return heartbeat;
	}

	public void setHeartbeatTime(MilliTime heartbeat){
		this.heartbeat = heartbeat;
	}

	public LongRunningTaskStatus getJobExecutionStatus(){
		return jobExecutionStatus;
	}

	public void setJobExecutionStatus(LongRunningTaskStatus longRunningTaskStatus){
		this.jobExecutionStatus = longRunningTaskStatus;
	}

	public String getTriggeredBy(){
		return triggeredBy;
	}

	public LongRunningTaskType getType(){
		return type;
	}

	public Long getNumItemsProcessed(){
		return numItemsProcessed;
	}

	public void setNumItemsProcessed(Long numItemsProcessed){
		this.numItemsProcessed = numItemsProcessed;
	}

	public String getLastItemProcessed(){
		return lastItemProcessed;
	}

	public void setLastItemProcessed(String lastItemProcessed){
		this.lastItemProcessed = lastItemProcessed;
	}

	public String getExceptionRecordId(){
		return exceptionRecordId;
	}

	public TaskTrackerDto toDto(){
		return new TaskTrackerDto(
				getKey().toDto(),
				type.persistentString,
				Optional.ofNullable(start)
						.map(MilliTime::toInstant)
						.orElse(null),
				Optional.ofNullable(finish)
						.map(MilliTime::toInstant)
						.orElse(null),
				Optional.ofNullable(heartbeat)
						.map(MilliTime::toInstant)
						.orElse(null),
				jobExecutionStatus.persistentString,
				triggeredBy,
				numItemsProcessed,
				lastItemProcessed,
				exceptionRecordId);
	}

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy