Please wait. This can take some minutes ...
Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance.
Project price only 1 $
You can buy this project and download/modify it how often you want.
com.arextest.schedule.progress.impl.UpdateResultProgressEventImpl Maven / Gradle / Ivy
package com.arextest.schedule.progress.impl;
import com.alibaba.fastjson2.util.DateUtils;
import com.arextest.common.cache.CacheProvider;
import com.arextest.schedule.bizlog.BizLogger;
import com.arextest.schedule.dao.mongodb.ReplayPlanActionRepository;
import com.arextest.schedule.dao.mongodb.ReplayPlanRepository;
import com.arextest.schedule.eventbus.PlanAutoRerunEvent;
import com.arextest.schedule.eventbus.PlanAutoReturnEventService;
import com.arextest.schedule.model.LogType;
import com.arextest.schedule.model.ReplayActionItem;
import com.arextest.schedule.model.ReplayPlan;
import com.arextest.schedule.model.ReplayStatusType;
import com.arextest.schedule.model.plan.PlanStageEnum;
import com.arextest.schedule.model.plan.ReplayPlanStageInfo;
import com.arextest.schedule.model.plan.StageBaseInfo;
import com.arextest.schedule.model.plan.StageStatusEnum;
import com.arextest.schedule.progress.ProgressEvent;
import com.arextest.schedule.service.MetricService;
import com.arextest.schedule.service.ReplayReportService;
import com.arextest.schedule.utils.RedisKeyBuildUtils;
import com.arextest.schedule.utils.StageUtils;
import com.arextest.web.model.contract.contracts.common.PlanStatistic;
import jakarta.annotation.Resource;
import java.util.Date;
import java.util.List;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Value;
/**
* @author jmo
* @since 2021/10/11
*/
@Slf4j
public class UpdateResultProgressEventImpl implements ProgressEvent {
public static final long DEFAULT_COUNT = 1L;
@Resource
private ReplayPlanActionRepository replayPlanActionRepository;
@Resource
private ReplayPlanRepository replayPlanRepository;
@Resource
private ReplayReportService replayReportService;
@Resource
private MetricService metricService;
@Resource
private CacheProvider redisCacheProvider;
@Resource
private PlanAutoReturnEventService planAutoReturnEventService;
@Value("${auto.rerun.threshold}")
private double autoRerunThreshold;
@Override
public void onReplayPlanReRunException(ReplayPlan plan, Throwable t) {
replayReportService.pushPlanStatus(plan.getId(), ReplayStatusType.FAIL_INTERRUPTED,
t.getMessage(), true);
redisCacheProvider.remove(RedisKeyBuildUtils.buildPlanRunningRedisKey(plan.getId()));
}
@Override
public void onReplayPlanCreated(ReplayPlan replayPlan) {
try {
onReplayPlanStageUpdate(replayPlan, PlanStageEnum.INIT_REPORT, StageStatusEnum.ONGOING,
System.currentTimeMillis(), null);
boolean success = replayReportService.initReportInfo(replayPlan);
StageStatusEnum stageStatusEnum = StageStatusEnum.success(success);
onReplayPlanStageUpdate(replayPlan, PlanStageEnum.INIT_REPORT, stageStatusEnum,
null, System.currentTimeMillis());
} catch (Throwable throwable) {
LOGGER.error("prepare load compare config error: {}, plan id:{}", throwable.getMessage(),
replayPlan.getId(), throwable);
}
}
@Override
public void onCompareConfigBeforeLoading(ReplayPlan replayPlan) {
onReplayPlanStageUpdate(replayPlan, PlanStageEnum.LOADING_CONFIG, StageStatusEnum.ONGOING,
System.currentTimeMillis(), null);
}
@Override
public void onCompareConfigLoaded(ReplayPlan replayPlan) {
onReplayPlanStageUpdate(replayPlan, PlanStageEnum.LOADING_CONFIG, StageStatusEnum.SUCCEEDED,
null, System.currentTimeMillis());
}
@Override
public boolean onBeforeReplayPlanFinish(ReplayPlan replayPlan) {
redisCacheProvider.remove(RedisKeyBuildUtils.buildPlanRunningRedisKey(replayPlan.getId()));
// only auto rerun once
if (replayPlan.isReRun()) {
return true;
}
if (autoRerunThreshold == 1) {
return true;
}
// When pass rate is more than the threshold
// wait 5 seconds to check the plan statistic
try {
Thread.sleep(5000);
} catch (InterruptedException e) {
LOGGER.error("sleep error:{}", e.getMessage(), e);
Thread.currentThread().interrupt();
}
PlanStatistic planStatistic = replayReportService.queryPlanStatistic(replayPlan.getId(),
replayPlan.getAppId());
if (planStatistic == null || planStatistic.getTotalCaseCount() == 0) {
LOGGER.error("query plan statistic error, plan id:{}", replayPlan.getId());
return true;
}
if (((double) planStatistic.getSuccessCaseCount()) / planStatistic.getTotalCaseCount()
< autoRerunThreshold) {
return true;
}
return false;
}
@Override
public void onReplayPlanAutoRerun(ReplayPlan replayPlan) {
PlanAutoRerunEvent event = new PlanAutoRerunEvent();
event.setPlanId(replayPlan.getId());
planAutoReturnEventService.postRerunAsyncEvent(event);
}
@Override
public void onReplayPlanFinish(ReplayPlan replayPlan, ReplayStatusType reason) {
replayPlan.setPlanFinishTime(new Date());
String planId = replayPlan.getId();
boolean result = replayPlanRepository.finish(planId);
LOGGER.info("update the replay plan finished, plan id:{} , result: {}", planId, result);
replayReportService.pushPlanStatus(planId, reason, null, replayPlan.isReRun());
recordPlanExecutionTime(replayPlan);
}
@Override
public void onReplayPlanInterrupt(ReplayPlan replayPlan, ReplayStatusType reason) {
replayPlan.setPlanFinishTime(new Date());
String planId = replayPlan.getId();
replayPlanRepository.finish(planId);
LOGGER.info("The plan was interrupted, plan id:{} ,appId: {} ", replayPlan.getId(),
replayPlan.getAppId());
metricService.recordCountEvent(LogType.PLAN_EXCEPTION_NUMBER.getValue(), replayPlan.getId(),
replayPlan.getAppId(), DEFAULT_COUNT);
replayReportService.pushPlanStatus(planId, reason, replayPlan.getErrorMessage(),
replayPlan.isReRun());
recordPlanExecutionTime(replayPlan);
redisCacheProvider.remove(RedisKeyBuildUtils.buildPlanRunningRedisKey(replayPlan.getId()));
BizLogger.recordPlanStatusChange(replayPlan, ReplayStatusType.FAIL_INTERRUPTED);
}
@Override
public void onReplayPlanTerminate(String replayId, String reason) {
replayPlanRepository.finish(replayId);
replayReportService.pushPlanStatus(replayId, ReplayStatusType.CANCELLED, reason, false);
redisCacheProvider.remove(RedisKeyBuildUtils.buildPlanRunningRedisKey(replayId));
}
@Override
public void onReplayPlanStageUpdate(ReplayPlan replayPlan, PlanStageEnum stageType,
StageStatusEnum stageStatus,
Long startTime, Long endTime, String msg) {
StageBaseInfo stageBaseInfo;
// if the plan is canceled, ignore the stage update
if (replayPlan.getPlanStatus() != null && replayPlan.getPlanStatus().isCanceled()) {
return;
}
if (stageType.isMainStage()) {
stageBaseInfo = findStage(replayPlan.getReplayPlanStageList(), stageType);
} else {
ReplayPlanStageInfo parentStage = findParentStage(replayPlan.getReplayPlanStageList(),
stageType);
updateParentStage(parentStage, stageType, stageStatus, startTime, endTime,
replayPlan.getOperator());
stageBaseInfo = findStage(parentStage.getSubStageInfoList(), stageType);
}
if (stageBaseInfo != null) {
updateStage(stageBaseInfo, stageType, stageStatus, startTime, endTime, msg,
replayPlan.getOperator());
}
replayPlan.setLastUpdateTime(System.currentTimeMillis());
}
@Override
public void onReplayPlanReRun(ReplayPlan replayPlan) {
replayReportService.pushPlanStatus(replayPlan.getId(), ReplayStatusType.RERUNNING, null,
replayPlan.isReRun());
redisCacheProvider.remove(RedisKeyBuildUtils.buildStopPlanRedisKey(replayPlan.getId()));
addReRunStage(replayPlan.getReplayPlanStageList());
replayPlanRepository.updateStage(replayPlan);
}
private StageBaseInfo findStage(List stageInfoList,
PlanStageEnum stageType) {
StageBaseInfo stageBaseInfo = null;
for (int i = stageInfoList.size() - 1; i >= 0; i--) {
ReplayPlanStageInfo stage = stageInfoList.get(i);
if (stageType.getCode() == stage.getStageType()) {
stageBaseInfo = stage;
break;
}
}
return stageBaseInfo;
}
private ReplayPlanStageInfo findParentStage(List stageInfoList,
PlanStageEnum stageType) {
ReplayPlanStageInfo parentStage = null;
PlanStageEnum parentStageEnum = PlanStageEnum.of(stageType.getParentStage());
for (int i = stageInfoList.size() - 1; i >= 0; i--) {
ReplayPlanStageInfo stage = stageInfoList.get(i);
if (parentStageEnum.getCode() == stage.getStageType()) {
parentStage = stage;
break;
}
}
return parentStage;
}
private void updateStage(StageBaseInfo stageBaseInfo, PlanStageEnum stageType,
StageStatusEnum stageStatus,
Long startTime, Long endTime, String msg, String operator) {
if (stageBaseInfo != null) {
if (msg == null) {
msg = String.format(StageUtils.MSG_FORMAT, stageType.name(), stageStatus);
}
stageBaseInfo.setStageType(stageType.getCode());
stageBaseInfo.setStageName(stageType.name());
if (stageStatus != null) {
stageBaseInfo.setStageStatus(stageStatus.getCode());
}
stageBaseInfo.setMsg(msg);
if (startTime != null) {
stageBaseInfo.setStartTime(startTime);
}
if (endTime != null) {
stageBaseInfo.setEndTime(endTime);
}
if (stageType == PlanStageEnum.RE_RUN) {
String startTimeStr =
stageBaseInfo.getStartTime() == null ? ""
: DateUtils.format(new Date(stageBaseInfo.getStartTime()));
String endTimeStr =
stageBaseInfo.getEndTime() == null ? ""
: DateUtils.format(new Date(stageBaseInfo.getEndTime()));
stageBaseInfo.setMsg(String.format(StageUtils.RUNNING_FORMAT, startTimeStr, endTimeStr, operator));
}
}
}
private void updateParentStage(StageBaseInfo parentStage, PlanStageEnum stageType,
StageStatusEnum stageStatus,
Long startTime, Long endTime, String operator) {
PlanStageEnum parentStageEnum = PlanStageEnum.of(parentStage.getStageType());
// when first subStage starts, start its parent stage
boolean firstSubStageOnGoing =
stageStatus == StageStatusEnum.ONGOING
&& parentStageEnum.getSubStageList().get(0) == stageType.getCode();
// when last subStage successes, end its parent stage
boolean lastSubStageSucceeded = stageStatus == StageStatusEnum.SUCCEEDED &&
parentStageEnum.getSubStageList().get(parentStageEnum.getSubStageList().size() - 1)
== stageType.getCode();
// when any subStage fails, fail its parent stage
if (firstSubStageOnGoing || lastSubStageSucceeded || stageStatus == StageStatusEnum.FAILED) {
updateStage(parentStage, parentStageEnum, stageStatus, startTime, endTime, null, operator);
}
}
private void addReRunStage(List stageInfoList) {
// reset stage after RUN&RERUN&CANCEL and add RERUN stage.
int addIndex = 0;
for (int index = 0; index < stageInfoList.size(); index++) {
if (stageInfoList.get(index).getStageType() == PlanStageEnum.RUN.getCode() ||
stageInfoList.get(index).getStageType() == PlanStageEnum.RE_RUN.getCode() ||
stageInfoList.get(index).getStageType() == PlanStageEnum.CANCEL.getCode()) {
addIndex = index + 1;
}
}
ReplayPlanStageInfo reRunStage = StageUtils.initEmptyStage(PlanStageEnum.RE_RUN);
stageInfoList.add(addIndex, reRunStage);
for (addIndex++; addIndex < stageInfoList.size(); addIndex++) {
StageUtils.resetStageStatus(stageInfoList.get(addIndex));
}
}
private void recordPlanExecutionTime(ReplayPlan replayPlan) {
Date planCreateTime = replayPlan.getPlanCreateTime();
long planFinishMills = replayPlan.getPlanFinishTime() == null ? System.currentTimeMillis()
: replayPlan.getPlanFinishTime().getTime();
if (planCreateTime != null) {
metricService.recordTimeEvent(LogType.PLAN_EXECUTION_TIME.getValue(), replayPlan.getId(),
replayPlan.getAppId(), null,
planFinishMills - planCreateTime.getTime());
} else {
LOGGER.warn("record plan execution time fail, plan create time is null, plan id :{}",
replayPlan.getId());
}
}
@Override
public void onActionBeforeSend(ReplayActionItem actionItem) {
actionItem.setReplayBeginTime(new Date());
}
@Override
public void onActionAfterSend(ReplayActionItem actionItem) {
}
@Override
public void onActionCaseLoaded(ReplayActionItem actionItem) {
if (actionItem.isEmpty()) {
return;
}
actionItem.setReplayStatus(ReplayStatusType.CASE_LOADED.getValue());
replayPlanActionRepository.update(actionItem);
LOGGER.info("update the replay action case count, action id:{} , size: {}", actionItem.getId(),
actionItem.getReplayCaseCount());
}
}