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

io.cdap.cdap.internal.app.program.ProgramStateWriterWithHeartBeat Maven / Gradle / Ivy

The newest version!
/*
 * Copyright © 2018 Cask Data, Inc.
 *
 * 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.cdap.cdap.internal.app.program;

import com.google.common.annotations.VisibleForTesting;
import com.google.gson.Gson;
import com.google.gson.GsonBuilder;
import io.cdap.cdap.app.runtime.Arguments;
import io.cdap.cdap.app.runtime.ProgramOptions;
import io.cdap.cdap.app.runtime.ProgramStateWriter;
import io.cdap.cdap.common.conf.CConfiguration;
import io.cdap.cdap.common.conf.Constants;
import io.cdap.cdap.internal.app.ApplicationSpecificationAdapter;
import io.cdap.cdap.internal.app.runtime.ProgramOptionConstants;
import io.cdap.cdap.internal.app.runtime.codec.ArgumentsCodec;
import io.cdap.cdap.internal.app.runtime.codec.ProgramOptionsCodec;
import io.cdap.cdap.messaging.spi.MessagingService;
import io.cdap.cdap.proto.Notification;
import io.cdap.cdap.proto.id.ProgramRunId;
import java.util.HashMap;
import java.util.Map;
import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.TimeUnit;
import javax.annotation.Nullable;
import org.apache.twill.common.Threads;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/**
 * Wrapper around {@link ProgramStateWriter} with additional hook to start a heartbeat thread on
 * running or resuming program state and stop the thread on completed/error/suspend state.
 */
public class ProgramStateWriterWithHeartBeat {

  private static final Logger LOG = LoggerFactory.getLogger(ProgramStateWriterWithHeartBeat.class);
  private static final Gson GSON =
      ApplicationSpecificationAdapter.addTypeAdapters(new GsonBuilder())
          .registerTypeAdapter(Arguments.class, new ArgumentsCodec())
          .registerTypeAdapter(ProgramOptions.class, new ProgramOptionsCodec()).create();
  private final long heartBeatIntervalSeconds;
  private final ProgramStateWriter programStateWriter;
  private final ProgramRunId programRunId;
  private final ProgramStatePublisher programStatePublisher;
  private ScheduledExecutorService scheduler;


  public ProgramStateWriterWithHeartBeat(ProgramRunId programRunId,
      ProgramStateWriter programStateWriter,
      MessagingService messagingService,
      CConfiguration cConf) {
    this(programRunId, programStateWriter,
        cConf.getLong(Constants.ProgramHeartbeat.HEARTBEAT_INTERVAL_SECONDS),
        new MessagingProgramStatePublisher(cConf, messagingService));
  }

  @VisibleForTesting
  ProgramStateWriterWithHeartBeat(ProgramRunId programRunId,
      ProgramStateWriter programStateWriter,
      long heartBeatIntervalSeconds,
      ProgramStatePublisher programStatePublisher) {
    this.programRunId = programRunId;
    this.programStateWriter = programStateWriter;
    this.heartBeatIntervalSeconds = heartBeatIntervalSeconds;
    this.programStatePublisher = programStatePublisher;
  }

  public void running(@Nullable String twillRunId) {
    programStateWriter.running(programRunId, twillRunId);
    scheduleHeartBeatThread();
  }

  public void completed() {
    stopHeartbeatThread();
    programStateWriter.completed(programRunId);
  }

  public void killed() {
    stopHeartbeatThread();
    programStateWriter.killed(programRunId);
  }

  public void error(Throwable failureCause) {
    stopHeartbeatThread();
    programStateWriter.error(programRunId, failureCause);
  }

  /**
   * If executor service isn't initialized or if its shutdown create a new executor service and
   * schedule a heartbeat thread
   */
  private void scheduleHeartBeatThread() {
    if (scheduler == null) {
      scheduler = Executors.newSingleThreadScheduledExecutor(
          Threads.createDaemonThreadFactory("program-heart-beat"));
      scheduler.scheduleAtFixedRate(
          () -> {
            Map properties = new HashMap<>();
            properties.put(ProgramOptionConstants.PROGRAM_RUN_ID, GSON.toJson(programRunId));
            properties.put(ProgramOptionConstants.HEART_BEAT_TIME,
                String.valueOf(System.currentTimeMillis()));
            // publish as heart_beat type, so it can be handled appropriately at receiver
            programStatePublisher.publish(Notification.Type.PROGRAM_HEART_BEAT, properties);
            LOG.trace("Sent heartbeat for program {}", programRunId);
          }, heartBeatIntervalSeconds,
          heartBeatIntervalSeconds, TimeUnit.SECONDS);
    }
  }

  private void stopHeartbeatThread() {
    if (scheduler != null) {
      scheduler.shutdownNow();
      scheduler = null;
    }
  }

  /**
   * This method is only used for testing
   *
   * @return true if the heart beat thread is active, false otherwise
   */
  @VisibleForTesting
  boolean isHeartBeatThreadAlive() {
    return scheduler != null && !scheduler.isShutdown();
  }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy