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

org.apache.dolphinscheduler.api.service.impl.DataAnalysisServiceImpl Maven / Gradle / Ivy

There is a newer version: 3.2.1
Show newest version
/*
 * Licensed to the Apache Software Foundation (ASF) under one or more
 * contributor license agreements.  See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The ASF licenses this file to You 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 org.apache.dolphinscheduler.api.service.impl;

import org.apache.commons.lang3.StringUtils;
import org.apache.commons.lang3.tuple.Pair;
import org.apache.dolphinscheduler.api.dto.CommandStateCount;
import org.apache.dolphinscheduler.api.dto.DefineUserDto;
import org.apache.dolphinscheduler.api.dto.TaskCountDto;
import org.apache.dolphinscheduler.api.enums.Status;
import org.apache.dolphinscheduler.api.service.DataAnalysisService;
import org.apache.dolphinscheduler.api.service.ProjectService;
import org.apache.dolphinscheduler.common.constants.Constants;
import org.apache.dolphinscheduler.common.enums.AuthorizationType;
import org.apache.dolphinscheduler.common.enums.CommandType;
import org.apache.dolphinscheduler.common.enums.UserType;
import org.apache.dolphinscheduler.common.utils.DateUtils;
import org.apache.dolphinscheduler.common.utils.TriFunction;
import org.apache.dolphinscheduler.dao.entity.CommandCount;
import org.apache.dolphinscheduler.dao.entity.DefinitionGroupByUser;
import org.apache.dolphinscheduler.dao.entity.ExecuteStatusCount;
import org.apache.dolphinscheduler.dao.entity.Project;
import org.apache.dolphinscheduler.dao.entity.User;
import org.apache.dolphinscheduler.dao.mapper.CommandMapper;
import org.apache.dolphinscheduler.dao.mapper.ErrorCommandMapper;
import org.apache.dolphinscheduler.dao.mapper.ProcessDefinitionMapper;
import org.apache.dolphinscheduler.dao.mapper.ProcessInstanceMapper;
import org.apache.dolphinscheduler.dao.mapper.ProjectMapper;
import org.apache.dolphinscheduler.dao.mapper.TaskInstanceMapper;
import org.apache.dolphinscheduler.plugin.task.api.enums.TaskExecutionStatus;
import org.apache.dolphinscheduler.service.process.ProcessService;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;

import java.util.ArrayList;
import java.util.Arrays;
import java.util.Date;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.Optional;
import java.util.Set;
import java.util.stream.Collectors;

import static org.apache.dolphinscheduler.api.constants.ApiFuncIdentificationConstant.PROJECT_OVERVIEW;

/**
 * data analysis service impl
 */
@Service
public class DataAnalysisServiceImpl extends BaseServiceImpl implements DataAnalysisService {

    private static final Logger logger = LoggerFactory.getLogger(DataAnalysisServiceImpl.class);

    @Autowired
    private ProjectMapper projectMapper;

    @Autowired
    private ProjectService projectService;

    @Autowired
    private ProcessInstanceMapper processInstanceMapper;

    @Autowired
    private ProcessDefinitionMapper processDefinitionMapper;

    @Autowired
    private CommandMapper commandMapper;

    @Autowired
    private ErrorCommandMapper errorCommandMapper;

    @Autowired
    private TaskInstanceMapper taskInstanceMapper;

    @Autowired
    private ProcessService processService;

    /**
     * statistical task instance status data
     *
     * @param loginUser   login user
     * @param projectCode project code
     * @param startDate   start date
     * @param endDate     end date
     * @return task state count data
     */
    @Override
    public Map countTaskStateByProject(User loginUser, long projectCode, String startDate,
                                                       String endDate) {

        return countStateByProject(
                loginUser,
                projectCode,
                startDate,
                endDate,
                this::countTaskInstanceAllStatesByProjectCodes);
    }

    /**
     * statistical process instance status data
     *
     * @param loginUser   login user
     * @param projectCode project code
     * @param startDate   start date
     * @param endDate     end date
     * @return process instance state count data
     */
    @Override
    public Map countProcessInstanceStateByProject(User loginUser, long projectCode, String startDate,
                                                                  String endDate) {
        Map result = this.countStateByProject(
                loginUser,
                projectCode,
                startDate,
                endDate,
                (start, end, projectCodes) -> this.processInstanceMapper.countInstanceStateByProjectCodes(start, end,
                        projectCodes));

        // process state count needs to remove state of forced success
        if (result.containsKey(Constants.STATUS) && result.get(Constants.STATUS).equals(Status.SUCCESS)) {
            ((TaskCountDto) result.get(Constants.DATA_LIST))
                    .removeStateFromCountList(TaskExecutionStatus.FORCED_SUCCESS);
        }
        return result;
    }

    /**
     * Wrapper function of counting process instance state and task state
     *
     * @param loginUser   login user
     * @param projectCode project code
     * @param startDate   start date
     * @param endDate     end date
     */
    private Map countStateByProject(User loginUser, long projectCode, String startDate, String endDate,
                                                    TriFunction> instanceStateCounter) {
        Map result = new HashMap<>();
        if (projectCode != 0) {
            Project project = projectMapper.queryByCode(projectCode);
            result = projectService.checkProjectAndAuth(loginUser, project, projectCode, PROJECT_OVERVIEW);
            if (result.get(Constants.STATUS) != Status.SUCCESS) {
                return result;
            }
        }

        Date start = null;
        Date end = null;
        if (!StringUtils.isEmpty(startDate) && !StringUtils.isEmpty(endDate)) {
            start = DateUtils.stringToDate(startDate);
            end = DateUtils.stringToDate(endDate);
            if (Objects.isNull(start) || Objects.isNull(end)) {
                putMsg(result, Status.REQUEST_PARAMS_NOT_VALID_ERROR, Constants.START_END_DATE);
                return result;
            }
        }
        Pair, Map> projectIds = getProjectIds(loginUser, result);
        if (projectIds.getRight() != null) {
            return projectIds.getRight();
        }
        Long[] projectCodeArray =
                projectCode == 0 ? getProjectCodesArrays(projectIds.getLeft()) : new Long[]{projectCode};
        List processInstanceStateCounts = new ArrayList<>();

        if (projectCodeArray.length != 0 || loginUser.getUserType() == UserType.ADMIN_USER) {
            processInstanceStateCounts = instanceStateCounter.apply(start, end, projectCodeArray);
        }

        if (processInstanceStateCounts != null) {
            TaskCountDto taskCountResult = new TaskCountDto(processInstanceStateCounts);
            result.put(Constants.DATA_LIST, taskCountResult);
            putMsg(result, Status.SUCCESS);
        }
        return result;
    }

    /**
     * statistics the process definition quantities of a certain person
     * 

* We only need projects which users have permission to see to determine whether the definition belongs to the user or not. * * @param loginUser login user * @param projectCode project code * @return definition count data */ @Override public Map countDefinitionByUser(User loginUser, long projectCode) { Map result = new HashMap<>(); if (projectCode != 0) { Project project = projectMapper.queryByCode(projectCode); result = projectService.checkProjectAndAuth(loginUser, project, projectCode, PROJECT_OVERVIEW); if (result.get(Constants.STATUS) != Status.SUCCESS) { return result; } } List defineGroupByUsers = new ArrayList<>(); Pair, Map> projectIds = getProjectIds(loginUser, result); if (projectIds.getRight() != null) { List emptyList = new ArrayList<>(); DefineUserDto dto = new DefineUserDto(emptyList); result.put(Constants.DATA_LIST, dto); putMsg(result, Status.SUCCESS); return result; } Long[] projectCodeArray = projectCode == 0 ? getProjectCodesArrays(projectIds.getLeft()) : new Long[]{projectCode}; if (projectCodeArray.length != 0 || loginUser.getUserType() == UserType.ADMIN_USER) { defineGroupByUsers = processDefinitionMapper.countDefinitionByProjectCodes(projectCodeArray); } DefineUserDto dto = new DefineUserDto(defineGroupByUsers); result.put(Constants.DATA_LIST, dto); putMsg(result, Status.SUCCESS); return result; } /** * statistical command status data * * @param loginUser login user * @return command state count data */ @Override public Map countCommandState(User loginUser) { Map result = new HashMap<>(); /** * find all the task lists in the project under the user * statistics based on task status execution, failure, completion, wait, total */ Date start = null; Date end = null; Pair, Map> projectIds = getProjectIds(loginUser, result); if (projectIds.getRight() != null) { List noData = Arrays.stream(CommandType.values()) .map(commandType -> new CommandStateCount(0, 0, commandType)).collect(Collectors.toList()); result.put(Constants.DATA_LIST, noData); putMsg(result, Status.SUCCESS); return result; } Long[] projectCodeArray = getProjectCodesArrays(projectIds.getLeft()); // count normal command state Map normalCountCommandCounts = commandMapper.countCommandState(start, end, projectCodeArray) .stream() .collect(Collectors.toMap(CommandCount::getCommandType, CommandCount::getCount)); // count error command state Map errorCommandCounts = errorCommandMapper.countCommandState(start, end, projectCodeArray) .stream() .collect(Collectors.toMap(CommandCount::getCommandType, CommandCount::getCount)); List list = Arrays.stream(CommandType.values()) .map(commandType -> new CommandStateCount( errorCommandCounts.getOrDefault(commandType, 0), normalCountCommandCounts.getOrDefault(commandType, 0), commandType)) .collect(Collectors.toList()); result.put(Constants.DATA_LIST, list); putMsg(result, Status.SUCCESS); return result; } private Pair, Map> getProjectIds(User loginUser, Map result) { Set projectIds = resourcePermissionCheckService .userOwnedResourceIdsAcquisition(AuthorizationType.PROJECTS, loginUser.getId(), logger); if (projectIds.isEmpty()) { List taskInstanceStateCounts = new ArrayList<>(); result.put(Constants.DATA_LIST, new TaskCountDto(taskInstanceStateCounts)); putMsg(result, Status.SUCCESS); return Pair.of(null, result); } return Pair.of(projectIds, null); } private Long[] getProjectCodesArrays(Set projectIds) { List projects = projectMapper.selectBatchIds(projectIds); List codeList = projects.stream().map(Project::getCode).collect(Collectors.toList()); Long[] projectCodeArray = new Long[codeList.size()]; codeList.toArray(projectCodeArray); return projectCodeArray; } /** * count queue state * * @return queue state count data */ @Override public Map countQueueState(User loginUser) { Map result = new HashMap<>(); // TODO need to add detail data info Map dataMap = new HashMap<>(); dataMap.put("taskQueue", 0); dataMap.put("taskKill", 0); result.put(Constants.DATA_LIST, dataMap); putMsg(result, Status.SUCCESS); return result; } @Override public List countTaskInstanceAllStatesByProjectCodes(Date startTime, Date endTime, Long[] projectCodes) { Optional> startTimeStates = Optional.ofNullable( this.taskInstanceMapper.countTaskInstanceStateByProjectCodes(startTime, endTime, projectCodes)); List allState = Arrays.stream(TaskExecutionStatus.values()).collect(Collectors.toList()); List needRecountState; if (startTimeStates.isPresent() && startTimeStates.get().size() != 0) { List instanceState = startTimeStates.get().stream().map(ExecuteStatusCount::getState).collect(Collectors.toList()); // value 0 state need to recount by submit time needRecountState = allState.stream().filter(ele -> !instanceState.contains(ele)).collect(Collectors.toList()); if (needRecountState.size() == 0) { return startTimeStates.get(); } } else { needRecountState = allState; } // use submit time to recount when 0 // if have any issues with this code, should change to specified states 0 8 9 17 not state count is 0 List recounts = this.taskInstanceMapper .countTaskInstanceStateByProjectCodesAndStatesBySubmitTime(startTime, endTime, projectCodes, needRecountState); startTimeStates.orElseGet(ArrayList::new).addAll(recounts); return startTimeStates.orElse(null); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy