com.dangdang.ddframe.job.cloud.scheduler.state.running.RunningService.java Source code

Java tutorial

Introduction

Here is the source code for com.dangdang.ddframe.job.cloud.scheduler.state.running.RunningService.java

Source

/*
 * Copyright 1999-2015 dangdang.com.
 * <p>
 * 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.
 * </p>
 */

package com.dangdang.ddframe.job.cloud.scheduler.state.running;

import com.dangdang.ddframe.job.cloud.scheduler.config.job.CloudJobConfiguration;
import com.dangdang.ddframe.job.cloud.scheduler.config.job.CloudJobConfigurationService;
import com.dangdang.ddframe.job.cloud.scheduler.config.job.CloudJobExecutionType;
import com.dangdang.ddframe.job.cloud.scheduler.mesos.MesosSlaveService;
import com.dangdang.ddframe.job.context.TaskContext;
import com.dangdang.ddframe.job.reg.base.CoordinatorRegistryCenter;
import com.google.common.base.Function;
import com.google.common.base.Optional;
import com.google.common.base.Predicate;
import com.google.common.collect.Iterators;
import com.google.common.collect.Lists;
import com.google.common.collect.Sets;
import com.google.common.util.concurrent.ThreadFactoryBuilder;
import com.google.gson.JsonArray;
import com.google.gson.JsonElement;
import lombok.Getter;
import lombok.RequiredArgsConstructor;

import java.util.Collection;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.CopyOnWriteArraySet;
import java.util.concurrent.Executors;

/**
 * ??.
 *
 * @author zhangliang
 */
@RequiredArgsConstructor
public final class RunningService {

    private static final int TASK_INITIAL_SIZE = 1024;

    // TODO JMX
    @Getter
    private static final ConcurrentHashMap<String, Set<TaskContext>> RUNNING_TASKS = new ConcurrentHashMap<>(
            TASK_INITIAL_SIZE);

    private static final ConcurrentHashMap<String, String> TASK_HOSTNAME_MAPPER = new ConcurrentHashMap<>(
            TASK_INITIAL_SIZE);

    private final CoordinatorRegistryCenter regCenter;

    private final CloudJobConfigurationService configurationService;

    private final MesosSlaveService mesosSlaveService;

    public RunningService(final CoordinatorRegistryCenter regCenter) {
        this.regCenter = regCenter;
        this.configurationService = new CloudJobConfigurationService(regCenter);
        this.mesosSlaveService = new MesosSlaveService();
    }

    /**
     * ??.
     */
    public void start() {
        clear();
        List<String> jobKeys = regCenter.getChildrenKeys(RunningNode.ROOT);
        for (String each : jobKeys) {
            if (!configurationService.load(each).isPresent()) {
                remove(each);
                continue;
            }
            RUNNING_TASKS.put(each,
                    Sets.newCopyOnWriteArraySet(
                            Lists.transform(regCenter.getChildrenKeys(RunningNode.getRunningJobNodePath(each)),
                                    new Function<String, TaskContext>() {

                                        @Override
                                        public TaskContext apply(final String input) {
                                            return TaskContext
                                                    .from(regCenter.get(RunningNode.getRunningTaskNodePath(
                                                            TaskContext.MetaInfo.from(input).toString())));
                                        }
                                    })));
        }
        if (RUNNING_TASKS.isEmpty()) {
            return;
        }
        Executors
                .newSingleThreadExecutor(
                        new ThreadFactoryBuilder().setDaemon(true).setNameFormat("Init-RunningService-%d").build())
                .execute(new Runnable() {
                    @Override
                    public void run() {
                        JsonArray slaves;
                        while ((slaves = mesosSlaveService.findAllSlaves()).size() < 1) {
                            try {
                                Thread.sleep(5000);
                            } catch (final InterruptedException ignored) {
                            }
                        }
                        for (Set<TaskContext> each : RUNNING_TASKS.values()) {
                            for (final TaskContext taskContext : each) {
                                Optional<JsonElement> slaveOptional = Iterators.tryFind(slaves.iterator(),
                                        new Predicate<JsonElement>() {
                                            @Override
                                            public boolean apply(final JsonElement input) {
                                                return input.getAsJsonObject().get("id").getAsString()
                                                        .equals(taskContext.getSlaveId());
                                            }
                                        });
                                if (slaveOptional.isPresent()) {
                                    TASK_HOSTNAME_MAPPER.putIfAbsent(taskContext.getId(),
                                            slaveOptional.get().getAsJsonObject().get("hostname").getAsString());
                                }
                            }
                        }
                    }
                });
    }

    /**
     * ??.
     * 
     * @param taskContext ?
     */
    public void add(final TaskContext taskContext) {
        if (!configurationService.load(taskContext.getMetaInfo().getJobName()).isPresent()) {
            return;
        }
        getRunningTasks(taskContext.getMetaInfo().getJobName()).add(taskContext);
        if (!isDaemon(taskContext.getMetaInfo().getJobName())) {
            return;
        }
        String runningTaskNodePath = RunningNode.getRunningTaskNodePath(taskContext.getMetaInfo().toString());
        if (!regCenter.isExisted(runningTaskNodePath)) {
            regCenter.persist(runningTaskNodePath, taskContext.getId());
        }
    }

    private boolean isDaemon(final String jobName) {
        Optional<CloudJobConfiguration> cloudJobConfigurationOptional = configurationService.load(jobName);
        return cloudJobConfigurationOptional.isPresent()
                && CloudJobExecutionType.DAEMON == cloudJobConfigurationOptional.get().getJobExecutionType();
    }

    /**
     * ?.
     * @param taskContext ?
     * @param isIdle ?
     */
    public void updateIdle(final TaskContext taskContext, final boolean isIdle) {
        synchronized (RUNNING_TASKS) {
            Optional<TaskContext> taskContextOptional = findTask(taskContext);
            if (taskContextOptional.isPresent()) {
                taskContextOptional.get().setIdle(isIdle);
            } else {
                add(taskContext);
            }
        }
    }

    private Optional<TaskContext> findTask(final TaskContext taskContext) {
        return Iterators.tryFind(getRunningTasks(taskContext.getMetaInfo().getJobName()).iterator(),
                new Predicate<TaskContext>() {
                    @Override
                    public boolean apply(final TaskContext input) {
                        return input.equals(taskContext);
                    }
                });
    }

    /**
     * ?.
     *
     * @param jobName ??
     */
    public void remove(final String jobName) {
        RUNNING_TASKS.remove(jobName);
        if (!isDaemonOrAbsent(jobName)) {
            return;
        }
        regCenter.remove(RunningNode.getRunningJobNodePath(jobName));
    }

    /**
     * ?.
     * 
     * @param taskContext ?
     */
    public void remove(final TaskContext taskContext) {
        getRunningTasks(taskContext.getMetaInfo().getJobName()).remove(taskContext);
        if (!isDaemonOrAbsent(taskContext.getMetaInfo().getJobName())) {
            return;
        }
        regCenter.remove(RunningNode.getRunningTaskNodePath(taskContext.getMetaInfo().toString()));
        String jobRootNode = RunningNode.getRunningJobNodePath(taskContext.getMetaInfo().getJobName());
        if (regCenter.isExisted(jobRootNode) && regCenter.getChildrenKeys(jobRootNode).isEmpty()) {
            regCenter.remove(jobRootNode);
        }
    }

    private boolean isDaemonOrAbsent(final String jobName) {
        Optional<CloudJobConfiguration> cloudJobConfigurationOptional = configurationService.load(jobName);
        return !cloudJobConfigurationOptional.isPresent()
                || CloudJobExecutionType.DAEMON == cloudJobConfigurationOptional.get().getJobExecutionType();
    }

    /**
     * ??.
     *
     * @param jobName ??
     * @return ??
     */
    public boolean isJobRunning(final String jobName) {
        return !getRunningTasks(jobName).isEmpty();
    }

    /**
     * ??.
     *
     * @param metaInfo ?
     * @return ??
     */
    public boolean isTaskRunning(final TaskContext.MetaInfo metaInfo) {
        for (TaskContext each : getRunningTasks(metaInfo.getJobName())) {
            if (each.getMetaInfo().equals(metaInfo)) {
                return true;
            }
        }
        return false;
    }

    /**
     * ???.
     *
     * @param jobName ??
     * @return ??
     */
    public Collection<TaskContext> getRunningTasks(final String jobName) {
        Set<TaskContext> taskContexts = new CopyOnWriteArraySet<>();
        Collection<TaskContext> result = RUNNING_TASKS.putIfAbsent(jobName, taskContexts);
        return null == result ? taskContexts : result;
    }

    /**
     * ??.
     *
     * @return ?
     */
    public Map<String, Set<TaskContext>> getAllRunningTasks() {
        Map<String, Set<TaskContext>> result = new HashMap<>(RUNNING_TASKS.size(), 1);
        result.putAll(RUNNING_TASKS);
        return result;
    }

    /**
     * ??.
     * 
     * @return ??
     */
    public Set<TaskContext> getAllRunningDaemonTasks() {
        List<String> jobKeys = regCenter.getChildrenKeys(RunningNode.ROOT);
        for (String each : jobKeys) {
            if (!RUNNING_TASKS.containsKey(each)) {
                remove(each);
            }
        }
        Set<TaskContext> result = Sets.newHashSet();
        for (Map.Entry<String, Set<TaskContext>> each : RUNNING_TASKS.entrySet()) {
            if (isDaemonOrAbsent(each.getKey())) {
                result.addAll(each.getValue());
            }
        }
        return result;
    }

    /**
     * ??.
     *
     * @param taskId 
     * @param hostname ??
     */
    public void addMapping(final String taskId, final String hostname) {
        TASK_HOSTNAME_MAPPER.putIfAbsent(taskId, hostname);
    }

    /**
     * ????.
     *
     * @param taskId 
     * @return ??
     */
    public String popMapping(final String taskId) {
        return TASK_HOSTNAME_MAPPER.remove(taskId);
    }

    /**
     * ????.
     *
     * @param taskId 
     * @return hostName ??
     */
    public String getHostNameByTaskId(final String taskId) {
        return TASK_HOSTNAME_MAPPER.get(taskId);
    }

    /**
     * ?zk?running.
     *
     * @param taskId 
     * @return running?
     */
    public boolean getRunningTaskInZookeeper(final String taskId) {
        List<String> jobKeys = regCenter.getChildrenKeys(RunningNode.ROOT);
        boolean taskExistFlag = false;
        if (0 == jobKeys.size()) {
            return false;
        }
        for (String each : jobKeys) {
            if (taskId.split("@")[0].equals(each)) {
                List<String> taskKeys = regCenter.getChildrenKeys(RunningNode.getRunningJobNodePath(each));
                if (0 == taskKeys.size()) {
                    return false;
                }
                for (String eachTask : taskKeys) {
                    if (taskId.contains(eachTask)) {
                        taskExistFlag = true;
                    }
                }
            }
        }
        return taskExistFlag;
    }

    /**
     * ???.
     */
    public void clear() {
        RUNNING_TASKS.clear();
        TASK_HOSTNAME_MAPPER.clear();
    }
}