org.apache.hive.spark.client.JobHandleImpl Maven / Gradle / Ivy
/*
* 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.hive.spark.client;
import java.io.Serializable;
import java.util.List;
import java.util.concurrent.CopyOnWriteArrayList;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException;
import com.google.common.base.Throwables;
import com.google.common.collect.Lists;
import io.netty.util.concurrent.Promise;
import org.apache.hive.spark.counter.SparkCounters;
/**
* A handle to a submitted job. Allows for monitoring and controlling of the running remote job.
*/
class JobHandleImpl implements JobHandle {
private final SparkClientImpl client;
private final String jobId;
private final MetricsCollection metrics;
private final Promise promise;
private final List sparkJobIds;
private final List listeners;
private volatile State state;
private volatile SparkCounters sparkCounters;
JobHandleImpl(SparkClientImpl client, Promise promise, String jobId) {
this.client = client;
this.jobId = jobId;
this.promise = promise;
this.listeners = Lists.newLinkedList();
this.metrics = new MetricsCollection();
this.sparkJobIds = new CopyOnWriteArrayList();
this.state = State.SENT;
this.sparkCounters = null;
}
/** Requests a running job to be cancelled. */
@Override
public boolean cancel(boolean mayInterrupt) {
if (changeState(State.CANCELLED)) {
client.cancel(jobId);
promise.cancel(mayInterrupt);
return true;
}
return false;
}
@Override
public T get() throws ExecutionException, InterruptedException {
return promise.get();
}
@Override
public T get(long timeout, TimeUnit unit)
throws ExecutionException, InterruptedException, TimeoutException {
return promise.get(timeout, unit);
}
@Override
public boolean isCancelled() {
return promise.isCancelled();
}
@Override
public boolean isDone() {
return promise.isDone();
}
/**
* The client job ID. This is unrelated to any Spark jobs that might be triggered by the
* submitted job.
*/
@Override
public String getClientJobId() {
return jobId;
}
/**
* A collection of metrics collected from the Spark jobs triggered by this job.
*
* To collect job metrics on the client, Spark jobs must be registered with JobContext::monitor()
* on the remote end.
*/
@Override
public MetricsCollection getMetrics() {
return metrics;
}
@Override
public List getSparkJobIds() {
return sparkJobIds;
}
@Override
public SparkCounters getSparkCounters() {
return sparkCounters;
}
@Override
public State getState() {
return state;
}
@Override
public void addListener(Listener l) {
synchronized (listeners) {
listeners.add(l);
// If current state is a final state, notify of Spark job IDs before notifying about the
// state transition.
if (state.ordinal() >= State.CANCELLED.ordinal()) {
for (Integer i : sparkJobIds) {
l.onSparkJobStarted(this, i);
}
}
fireStateChange(state, l);
// Otherwise, notify about Spark jobs after the state notification.
if (state.ordinal() < State.CANCELLED.ordinal()) {
for (Integer i : sparkJobIds) {
l.onSparkJobStarted(this, i);
}
}
}
}
public void setSparkCounters(SparkCounters sparkCounters) {
this.sparkCounters = sparkCounters;
}
@SuppressWarnings("unchecked")
void setSuccess(Object result) {
// The synchronization here is not necessary, but tests depend on it.
synchronized (listeners) {
promise.setSuccess((T) result);
changeState(State.SUCCEEDED);
}
}
void setFailure(Throwable error) {
// The synchronization here is not necessary, but tests depend on it.
synchronized (listeners) {
promise.setFailure(error);
changeState(State.FAILED);
}
}
/**
* Changes the state of this job handle, making sure that illegal state transitions are ignored.
* Fires events appropriately.
*
* As a rule, state transitions can only occur if the current state is "higher" than the current
* state (i.e., has a higher ordinal number) and is not a "final" state. "Final" states are
* CANCELLED, FAILED and SUCCEEDED, defined here in the code as having an ordinal number higher
* than the CANCELLED enum constant.
*/
boolean changeState(State newState) {
synchronized (listeners) {
if (newState.ordinal() > state.ordinal() && state.ordinal() < State.CANCELLED.ordinal()) {
state = newState;
for (Listener l : listeners) {
fireStateChange(newState, l);
}
return true;
}
return false;
}
}
void addSparkJobId(int sparkJobId) {
synchronized (listeners) {
sparkJobIds.add(sparkJobId);
for (Listener l : listeners) {
l.onSparkJobStarted(this, sparkJobId);
}
}
}
private void fireStateChange(State s, Listener l) {
switch (s) {
case SENT:
break;
case QUEUED:
l.onJobQueued(this);
break;
case STARTED:
l.onJobStarted(this);
break;
case CANCELLED:
l.onJobCancelled(this);
break;
case FAILED:
l.onJobFailed(this, promise.cause());
break;
case SUCCEEDED:
try {
l.onJobSucceeded(this, promise.get());
} catch (Exception e) {
// Shouldn't really happen.
throw new IllegalStateException(e);
}
break;
default:
throw new IllegalStateException();
}
}
/** Last attempt at preventing stray jobs from accumulating in SparkClientImpl. */
@Override
protected void finalize() {
if (!isDone()) {
cancel(true);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy