| | |
| | | import org.springframework.stereotype.Service; |
| | | import org.springframework.transaction.annotation.Transactional; |
| | | |
| | | import java.math.BigDecimal; |
| | | import java.math.RoundingMode; |
| | | import java.text.SimpleDateFormat; |
| | | import java.util.*; |
| | | import java.util.stream.Collectors; |
| | |
| | | WrkStsType.SETTLE_INBOUND.sts, |
| | | WrkStsType.COMPLETE_OUTBOUND.sts, |
| | | WrkStsType.SETTLE_OUTBOUND.sts, |
| | | WrkStsType.COMPLETE_LOC_MOVE.sts |
| | | WrkStsType.COMPLETE_LOC_MOVE.sts, |
| | | WrkStsType.COMPLETE_CRN_MOVE.sts |
| | | ); |
| | | private static final String MODE_TASK = "TASK"; |
| | | private static final String MODE_TIME = "TIME"; |
| | |
| | | private final BasCrnpErrLogService basCrnpErrLogService; |
| | | private final BasDualCrnpErrLogService basDualCrnpErrLogService; |
| | | private final BasRgvErrLogService basRgvErrLogService; |
| | | private final BasStationErrLogService basStationErrLogService; |
| | | private final BasStationService basStationService; |
| | | private final BasWrkStatusService basWrkStatusService; |
| | | private final WrkMastService wrkMastService; |
| | |
| | | BasCrnpErrLogService basCrnpErrLogService, |
| | | BasDualCrnpErrLogService basDualCrnpErrLogService, |
| | | BasRgvErrLogService basRgvErrLogService, |
| | | BasStationErrLogService basStationErrLogService, |
| | | BasStationService basStationService, |
| | | BasWrkStatusService basWrkStatusService, |
| | | WrkMastService wrkMastService) { |
| | |
| | | this.basCrnpErrLogService = basCrnpErrLogService; |
| | | this.basDualCrnpErrLogService = basDualCrnpErrLogService; |
| | | this.basRgvErrLogService = basRgvErrLogService; |
| | | this.basStationErrLogService = basStationErrLogService; |
| | | this.basStationService = basStationService; |
| | | this.basWrkStatusService = basWrkStatusService; |
| | | this.wrkMastService = wrkMastService; |
| | |
| | | entity.setDualCrnFaultDurationMs(defaultLong(entity.getDualCrnFaultDurationMs())); |
| | | entity.setRgvFaultCount(defaultInt(entity.getRgvFaultCount())); |
| | | entity.setRgvFaultDurationMs(defaultLong(entity.getRgvFaultDurationMs())); |
| | | entity.setStationFaultCount(defaultInt(entity.getStationFaultCount())); |
| | | entity.setStationFaultDurationMs(defaultLong(entity.getStationFaultDurationMs())); |
| | | entity.setMetricCompleteness(METRIC_PARTIAL); |
| | | entity.setUpdateTime(now); |
| | | this.saveOrUpdate(entity); |
| | |
| | | entity.setRgvNo(wrkMast.getRgvNo()); |
| | | entity.setFinalWrkSts(wrkMast.getWrkSts()); |
| | | entity.setUpdateTime(time); |
| | | if (Objects.equals(wrkMast.getIoType(), WrkIoType.LOC_MOVE.id) && entity.getStationDurationMs() == null) { |
| | | if ((Objects.equals(wrkMast.getIoType(), WrkIoType.LOC_MOVE.id) |
| | | || Objects.equals(wrkMast.getIoType(), WrkIoType.CRN_MOVE.id)) |
| | | && entity.getStationDurationMs() == null) { |
| | | entity.setStationDurationMs(0L); |
| | | } |
| | | this.updateById(entity); |
| | |
| | | if (entity.getAppeTime() != null) { |
| | | entity.setTotalDurationMs(durationMs(entity.getAppeTime(), time)); |
| | | } |
| | | if (Objects.equals(wrkMast.getIoType(), WrkIoType.LOC_MOVE.id) && entity.getStationDurationMs() == null) { |
| | | if ((Objects.equals(wrkMast.getIoType(), WrkIoType.LOC_MOVE.id) |
| | | || Objects.equals(wrkMast.getIoType(), WrkIoType.CRN_MOVE.id)) |
| | | && entity.getStationDurationMs() == null) { |
| | | entity.setStationDurationMs(0L); |
| | | } |
| | | FaultSummary faultSummary = buildFaultSummary(wrkMast.getWrkNo(), time); |
| | |
| | | entity.setDualCrnFaultDurationMs(faultSummary.dualDurationMs); |
| | | entity.setRgvFaultCount(faultSummary.rgvCount); |
| | | entity.setRgvFaultDurationMs(faultSummary.rgvDurationMs); |
| | | entity.setStationFaultCount(faultSummary.stationCount); |
| | | entity.setStationFaultDurationMs(faultSummary.stationDurationMs); |
| | | entity.setMetricCompleteness(resolveMetricCompleteness(wrkMast, entity)); |
| | | entity.setUpdateTime(time); |
| | | this.saveOrUpdate(entity); |
| | |
| | | ioTypes.add(option("1", "IN", "入库", WrkIoType.IN.id)); |
| | | ioTypes.add(option("2", "OUT", "出库", WrkIoType.OUT.id)); |
| | | ioTypes.add(option("3", "LOC_MOVE", "移库", WrkIoType.LOC_MOVE.id)); |
| | | ioTypes.add(option("4", "CRN_MOVE", "堆垛机移动", WrkIoType.CRN_MOVE.id)); |
| | | List<Map<String, Object>> timeFields = new ArrayList<>(); |
| | | timeFields.add(option(TIME_FIELD_FINISH, TIME_FIELD_FINISH, "完成时间", TIME_FIELD_FINISH)); |
| | | timeFields.add(option(TIME_FIELD_APPE, TIME_FIELD_APPE, "创建时间", TIME_FIELD_APPE)); |
| | |
| | | if (TIME_FIELD_APPE.equalsIgnoreCase(request.getString("timeField"))) { |
| | | timeField = TIME_FIELD_APPE; |
| | | } |
| | | return buildAnalysisResult(list, mode, timeField, request); |
| | | return buildAnalysisResult(list, timeField); |
| | | } |
| | | |
| | | private QueryWrapper<WrkMastLog> buildHistoryLogWrapper(Map<String, Object> param) { |
| | |
| | | wrapper.in("wrk_no", wrkNos); |
| | | } |
| | | |
| | | private Map<String, Object> buildAnalysisResult(List<WrkAnalysis> list, String mode, String timeField, JSONObject request) { |
| | | private Map<String, Object> buildAnalysisResult(List<WrkAnalysis> list, String timeField) { |
| | | Map<String, Object> result = new LinkedHashMap<>(); |
| | | Map<String, Object> summary = new LinkedHashMap<>(); |
| | | Date taskStartTime = list.stream() |
| | | .map(WrkAnalysis::getAppeTime) |
| | | .filter(Objects::nonNull) |
| | | .min(Date::compareTo) |
| | | .orElse(null); |
| | | Date taskEndTime = list.stream() |
| | | .map(WrkAnalysis::getFinishTime) |
| | | .filter(Objects::nonNull) |
| | | .max(Date::compareTo) |
| | | .orElse(null); |
| | | summary.put("taskCount", list.size()); |
| | | summary.put("taskStartTime", taskStartTime); |
| | | summary.put("taskStartTime$", formatDate(taskStartTime)); |
| | | summary.put("taskEndTime", taskEndTime); |
| | | summary.put("taskEndTime$", formatDate(taskEndTime)); |
| | | summary.put("taskDurationMs", taskStartTime == null || taskEndTime == null ? null : durationMs(taskStartTime, taskEndTime)); |
| | | summary.put("avgTaskBeatDurationMs", list.isEmpty() || taskStartTime == null || taskEndTime == null |
| | | ? null |
| | | : durationMs(taskStartTime, taskEndTime) / list.size()); |
| | | summary.put("avgTaskPerHour", calculateAvgTaskPerHour(list.size(), summary.get("taskDurationMs"))); |
| | | summary.put("avgTotalDurationMs", average(list, item -> item.getTotalDurationMs() != null, WrkAnalysis::getTotalDurationMs)); |
| | | summary.put("avgStationDurationMs", average(list, item -> !METRIC_PARTIAL.equals(item.getMetricCompleteness()) && item.getStationDurationMs() != null, WrkAnalysis::getStationDurationMs)); |
| | | summary.put("avgCraneDurationMs", average(list, item -> !METRIC_PARTIAL.equals(item.getMetricCompleteness()) && item.getCraneDurationMs() != null, WrkAnalysis::getCraneDurationMs)); |
| | |
| | | summary.put("partialTaskCount", list.stream().filter(item -> METRIC_PARTIAL.equals(item.getMetricCompleteness())).count()); |
| | | result.put("summary", summary); |
| | | result.put("durationCompare", buildDurationCompare(list)); |
| | | result.put("trend", buildTrend(list, mode, timeField, request)); |
| | | result.put("trend", buildTrend(list, timeField)); |
| | | result.put("faultPie", buildFaultPie(list)); |
| | | result.put("faultDuration", buildFaultDuration(list)); |
| | | result.put("detail", buildDetailRows(list)); |
| | |
| | | .collect(Collectors.toList()); |
| | | } |
| | | |
| | | private List<Map<String, Object>> buildTrend(List<WrkAnalysis> list, String mode, String timeField, JSONObject request) { |
| | | private List<Map<String, Object>> buildTrend(List<WrkAnalysis> list, String timeField) { |
| | | List<Map<String, Object>> trend = new ArrayList<>(); |
| | | if (list.isEmpty()) { |
| | | return trend; |
| | | } |
| | | long startMs = Long.MAX_VALUE; |
| | | long endMs = Long.MIN_VALUE; |
| | | if (MODE_TIME.equals(mode)) { |
| | | Long reqStart = parseLong(request.get("startTime")); |
| | | Long reqEnd = parseLong(request.get("endTime")); |
| | | if (reqStart != null && reqEnd != null) { |
| | | startMs = reqStart; |
| | | endMs = reqEnd; |
| | | } |
| | | } |
| | | if (startMs == Long.MAX_VALUE || endMs == Long.MIN_VALUE) { |
| | | for (WrkAnalysis item : list) { |
| | | Date date = resolveBucketTime(item, timeField); |
| | |
| | | if (startMs == Long.MAX_VALUE || endMs == Long.MIN_VALUE) { |
| | | return trend; |
| | | } |
| | | boolean hourly = endMs - startMs <= 72L * 60L * 60L * 1000L; |
| | | SimpleDateFormat keyFormat = new SimpleDateFormat(hourly ? "yyyy-MM-dd HH:00" : "yyyy-MM-dd"); |
| | | TrendBucketType bucketType = resolveTrendBucketType(startMs, endMs); |
| | | SimpleDateFormat keyFormat = new SimpleDateFormat(resolveTrendBucketPattern(bucketType)); |
| | | keyFormat.setTimeZone(TimeZone.getDefault()); |
| | | Map<String, BucketAccumulator> bucketMap = new LinkedHashMap<>(); |
| | | Map<Long, BucketAccumulator> bucketMap = new LinkedHashMap<>(); |
| | | List<WrkAnalysis> sorted = new ArrayList<>(list); |
| | | sorted.sort(Comparator.comparing(item -> resolveBucketTime(item, timeField), Comparator.nullsLast(Date::compareTo))); |
| | | for (WrkAnalysis item : sorted) { |
| | |
| | | if (bucketTime == null) { |
| | | continue; |
| | | } |
| | | String key = keyFormat.format(bucketTime); |
| | | Date bucketStart = truncateBucketTime(bucketTime, bucketType); |
| | | long key = bucketStart.getTime(); |
| | | BucketAccumulator accumulator = bucketMap.computeIfAbsent(key, k -> new BucketAccumulator()); |
| | | accumulator.taskCount++; |
| | | if (item.getTotalDurationMs() != null) { |
| | |
| | | accumulator.craneDurationCount++; |
| | | } |
| | | } |
| | | for (Map.Entry<String, BucketAccumulator> entry : bucketMap.entrySet()) { |
| | | for (Map.Entry<Long, BucketAccumulator> entry : bucketMap.entrySet()) { |
| | | Map<String, Object> item = new LinkedHashMap<>(); |
| | | item.put("bucketLabel", entry.getKey()); |
| | | item.put("bucketLabel", keyFormat.format(new Date(entry.getKey()))); |
| | | item.put("taskCount", entry.getValue().taskCount); |
| | | item.put("avgTotalDurationMs", entry.getValue().totalDurationCount == 0 ? null : entry.getValue().totalDurationMs / entry.getValue().totalDurationCount); |
| | | item.put("avgStationDurationMs", entry.getValue().stationDurationCount == 0 ? null : entry.getValue().stationDurationMs / entry.getValue().stationDurationCount); |
| | |
| | | trend.add(item); |
| | | } |
| | | return trend; |
| | | } |
| | | |
| | | private TrendBucketType resolveTrendBucketType(long startMs, long endMs) { |
| | | long spanMs = Math.max(0L, endMs - startMs); |
| | | if (spanMs <= 6L * 60L * 60L * 1000L) { |
| | | return TrendBucketType.MINUTE; |
| | | } |
| | | if (spanMs <= 72L * 60L * 60L * 1000L) { |
| | | return TrendBucketType.HOUR; |
| | | } |
| | | return TrendBucketType.DAY; |
| | | } |
| | | |
| | | private String resolveTrendBucketPattern(TrendBucketType bucketType) { |
| | | if (bucketType == TrendBucketType.MINUTE) { |
| | | return "yyyy-MM-dd HH:mm"; |
| | | } |
| | | if (bucketType == TrendBucketType.HOUR) { |
| | | return "yyyy-MM-dd HH:00"; |
| | | } |
| | | return "yyyy-MM-dd"; |
| | | } |
| | | |
| | | private Date truncateBucketTime(Date time, TrendBucketType bucketType) { |
| | | Calendar calendar = Calendar.getInstance(); |
| | | calendar.setTime(time); |
| | | if (bucketType == TrendBucketType.DAY) { |
| | | calendar.set(Calendar.HOUR_OF_DAY, 0); |
| | | } |
| | | if (bucketType == TrendBucketType.DAY || bucketType == TrendBucketType.HOUR) { |
| | | calendar.set(Calendar.MINUTE, 0); |
| | | } |
| | | calendar.set(Calendar.SECOND, 0); |
| | | calendar.set(Calendar.MILLISECOND, 0); |
| | | return calendar.getTime(); |
| | | } |
| | | |
| | | private List<Map<String, Object>> buildFaultPie(List<WrkAnalysis> list) { |
| | |
| | | } |
| | | |
| | | private List<Map<String, Object>> buildFaultDuration(List<WrkAnalysis> list) { |
| | | Map<String, Long> durationMap = new LinkedHashMap<>(); |
| | | for (WrkAnalysis item : list) { |
| | | addDeviceFaultDuration(durationMap, "单堆垛机", item.getCrnNo(), item.getCrnFaultDurationMs()); |
| | | addDeviceFaultDuration(durationMap, "双工位堆垛机", item.getDualCrnNo(), item.getDualCrnFaultDurationMs()); |
| | | addDeviceFaultDuration(durationMap, "RGV", item.getRgvNo(), item.getRgvFaultDurationMs()); |
| | | addDeviceFaultDuration(durationMap, "输送站点", null, item.getStationFaultDurationMs()); |
| | | } |
| | | List<Map<String, Object>> result = new ArrayList<>(); |
| | | result.add(slice("单堆垛机", list.stream().map(WrkAnalysis::getCrnFaultDurationMs).filter(Objects::nonNull).reduce(0L, Long::sum))); |
| | | result.add(slice("双工位堆垛机", list.stream().map(WrkAnalysis::getDualCrnFaultDurationMs).filter(Objects::nonNull).reduce(0L, Long::sum))); |
| | | result.add(slice("RGV", list.stream().map(WrkAnalysis::getRgvFaultDurationMs).filter(Objects::nonNull).reduce(0L, Long::sum))); |
| | | durationMap.forEach((name, durationMs) -> result.add(slice(name, durationMs))); |
| | | return result; |
| | | } |
| | | |
| | | private void addDeviceFaultDuration(Map<String, Long> durationMap, String deviceLabel, Integer deviceNo, Long durationMs) { |
| | | long value = defaultLong(durationMs); |
| | | if (value <= 0L) { |
| | | return; |
| | | } |
| | | String key = deviceNo == null ? deviceLabel : deviceLabel + deviceNo; |
| | | durationMap.merge(key, value, Long::sum); |
| | | } |
| | | |
| | | private List<Map<String, Object>> buildDetailRows(List<WrkAnalysis> list) { |
| | |
| | | target.put("crnFaultDurationMs", analysis == null ? 0L : defaultLong(analysis.getCrnFaultDurationMs())); |
| | | target.put("dualCrnFaultDurationMs", analysis == null ? 0L : defaultLong(analysis.getDualCrnFaultDurationMs())); |
| | | target.put("rgvFaultDurationMs", analysis == null ? 0L : defaultLong(analysis.getRgvFaultDurationMs())); |
| | | target.put("stationFaultDurationMs", analysis == null ? 0L : defaultLong(analysis.getStationFaultDurationMs())); |
| | | target.put("metricCompleteness", analysis == null ? METRIC_PARTIAL : analysis.getMetricCompleteness()); |
| | | } |
| | | |
| | |
| | | List<BasCrnpErrLog> crnList = basCrnpErrLogService.list(new QueryWrapper<BasCrnpErrLog>().eq("wrk_no", wrkNo)); |
| | | List<BasDualCrnpErrLog> dualList = basDualCrnpErrLogService.list(new QueryWrapper<BasDualCrnpErrLog>().eq("wrk_no", wrkNo)); |
| | | List<BasRgvErrLog> rgvList = basRgvErrLogService.list(new QueryWrapper<BasRgvErrLog>().eq("task_no", wrkNo)); |
| | | List<BasStationErrLog> stationList = basStationErrLogService.list(new QueryWrapper<BasStationErrLog>().eq("wrk_no", wrkNo)); |
| | | summary.crnCount = crnList.size(); |
| | | summary.crnDurationMs = durationMs(crnList, finishTime); |
| | | summary.dualCount = dualList.size(); |
| | | summary.dualDurationMs = durationMs(dualList, finishTime); |
| | | summary.rgvCount = rgvList.size(); |
| | | summary.rgvDurationMs = durationMs(rgvList, finishTime); |
| | | summary.totalCount = summary.crnCount + summary.dualCount + summary.rgvCount; |
| | | summary.totalDurationMs = summary.crnDurationMs + summary.dualDurationMs + summary.rgvDurationMs; |
| | | summary.stationCount = stationList.size(); |
| | | summary.stationDurationMs = durationMs(stationList, finishTime); |
| | | summary.totalCount = summary.crnCount + summary.dualCount + summary.rgvCount + summary.stationCount; |
| | | summary.totalDurationMs = summary.crnDurationMs + summary.dualDurationMs + summary.rgvDurationMs + summary.stationDurationMs; |
| | | summary.hasFault = summary.totalCount > 0 ? 1 : 0; |
| | | return summary; |
| | | } |
| | |
| | | } else if (item instanceof BasRgvErrLog) { |
| | | startTime = ((BasRgvErrLog) item).getStartTime(); |
| | | endTime = ((BasRgvErrLog) item).getEndTime(); |
| | | } else if (item instanceof BasStationErrLog) { |
| | | startTime = ((BasStationErrLog) item).getStartTime(); |
| | | endTime = ((BasStationErrLog) item).getEndTime(); |
| | | } |
| | | if (startTime == null) { |
| | | continue; |
| | |
| | | if (Objects.equals(ioType, WrkIoType.LOC_MOVE.id)) { |
| | | return "移库"; |
| | | } |
| | | if (Objects.equals(ioType, WrkIoType.CRN_MOVE.id)) { |
| | | return "堆垛机移动"; |
| | | } |
| | | return String.valueOf(ioType); |
| | | } |
| | | |
| | |
| | | count++; |
| | | } |
| | | return count == 0 ? null : total / count; |
| | | } |
| | | |
| | | private Double calculateAvgTaskPerHour(int taskCount, Object taskDurationMsValue) { |
| | | if (taskCount <= 0 || !(taskDurationMsValue instanceof Number)) { |
| | | return null; |
| | | } |
| | | long taskDurationMs = ((Number) taskDurationMsValue).longValue(); |
| | | if (taskDurationMs <= 0L) { |
| | | return null; |
| | | } |
| | | return BigDecimal.valueOf(taskCount * 3600000D / taskDurationMs) |
| | | .setScale(2, RoundingMode.HALF_UP) |
| | | .doubleValue(); |
| | | } |
| | | |
| | | private void applyRange(QueryWrapper<WrkMastLog> wrapper, String column, String rawValue) { |
| | |
| | | private long dualDurationMs; |
| | | private int rgvCount; |
| | | private long rgvDurationMs; |
| | | private int stationCount; |
| | | private long stationDurationMs; |
| | | } |
| | | |
| | | private static class BucketAccumulator { |
| | |
| | | private long craneDurationCount; |
| | | } |
| | | |
| | | private enum TrendBucketType { |
| | | MINUTE, |
| | | HOUR, |
| | | DAY |
| | | } |
| | | |
| | | } |