/*
* 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 gobblin.runtime.local;
import java.io.IOException;
import java.util.Iterator;
import java.util.List;
import java.util.Properties;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import com.google.common.base.Function;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.Iterators;
import com.google.common.collect.Lists;
import com.google.common.util.concurrent.ServiceManager;
import lombok.extern.slf4j.Slf4j;
import gobblin.broker.gobblin_scopes.GobblinScopeTypes;
import gobblin.broker.iface.SharedResourcesBroker;
import gobblin.metrics.Tag;
import gobblin.metrics.event.TimingEvent;
import gobblin.runtime.AbstractJobLauncher;
import gobblin.runtime.GobblinMultiTaskAttempt;
import gobblin.runtime.JobState;
import gobblin.runtime.TaskExecutor;
import gobblin.runtime.TaskStateTracker;
import gobblin.runtime.api.Configurable;
import gobblin.runtime.api.JobSpec;
import gobblin.source.workunit.WorkUnit;
import gobblin.util.JobConfigurationUtils;
import gobblin.runtime.util.MultiWorkUnitUnpackingIterator;
import gobblin.source.workunit.WorkUnitStream;
/**
* An implementation of {@link gobblin.runtime.JobLauncher} for launching and running jobs
* locally on a single node.
*
* @author Yinan Li
*/
@Slf4j
public class LocalJobLauncher extends AbstractJobLauncher {
private static final Logger LOG = LoggerFactory.getLogger(LocalJobLauncher.class);
private final TaskExecutor taskExecutor;
private final TaskStateTracker taskStateTracker;
// Service manager to manage dependent services
private final ServiceManager serviceManager;
public LocalJobLauncher(Properties jobProps) throws Exception {
this(jobProps, null);
}
public LocalJobLauncher(Properties jobProps, SharedResourcesBroker<GobblinScopeTypes> instanceBroker) throws Exception {
super(jobProps, ImmutableList.<Tag<?>> of(), instanceBroker);
log.debug("Local job launched with properties: {}", jobProps);
TimingEvent jobLocalSetupTimer = this.eventSubmitter.getTimingEvent(TimingEvent.RunJobTimings.JOB_LOCAL_SETUP);
this.taskExecutor = new TaskExecutor(jobProps);
this.taskStateTracker =
new LocalTaskStateTracker(jobProps, this.jobContext.getJobState(), this.taskExecutor, this.eventBus);
this.serviceManager = new ServiceManager(Lists.newArrayList(
// The order matters due to dependencies between services
this.taskExecutor, this.taskStateTracker));
// Start all dependent services
this.serviceManager.startAsync().awaitHealthy(5, TimeUnit.SECONDS);
startCancellationExecutor();
jobLocalSetupTimer.stop();
}
public LocalJobLauncher(Configurable instanceConf, JobSpec jobSpec) throws Exception {
this(JobConfigurationUtils.combineSysAndJobProperties(instanceConf.getConfigAsProperties(),
jobSpec.getConfigAsProperties()));
}
@Override
public void close() throws IOException {
try {
// Stop all dependent services
this.serviceManager.stopAsync().awaitStopped(5, TimeUnit.SECONDS);
} catch (TimeoutException te) {
LOG.warn("Timed out while waiting for the service manager to be stopped", te);
} finally {
super.close();
}
}
@Override
protected void runWorkUnits(List<WorkUnit> workUnits) throws Exception {
// This should never happen
throw new UnsupportedOperationException();
}
@Override
protected void runWorkUnitStream(WorkUnitStream workUnitStream) throws Exception {
String jobId = this.jobContext.getJobId();
final JobState jobState = this.jobContext.getJobState();
Iterator<WorkUnit> workUnitIterator = workUnitStream.getWorkUnits();
if (!workUnitIterator.hasNext()) {
LOG.warn("No work units to run");
return;
}
TimingEvent workUnitsRunTimer = this.eventSubmitter.getTimingEvent(TimingEvent.RunJobTimings.WORK_UNITS_RUN);
Iterator<WorkUnit> flattenedWorkUnits = new MultiWorkUnitUnpackingIterator(workUnitStream.getWorkUnits());
Iterator<WorkUnit> workUnitsWithJobState = Iterators.transform(flattenedWorkUnits, new Function<WorkUnit, WorkUnit>() {
@Override
public WorkUnit apply(WorkUnit workUnit) {
workUnit.addAllIfNotExist(jobState);
return workUnit;
}
});
GobblinMultiTaskAttempt.runWorkUnits(this.jobContext, workUnitsWithJobState, this.taskStateTracker,
this.taskExecutor, GobblinMultiTaskAttempt.CommitPolicy.IMMEDIATE);
if (this.cancellationRequested) {
// Wait for the cancellation execution if it has been requested
synchronized (this.cancellationExecution) {
if (this.cancellationExecuted) {
return;
}
}
}
workUnitsRunTimer.stop();
LOG.info(String.format("All tasks of job %s have completed", jobId));
if (jobState.getState() == JobState.RunningState.RUNNING) {
jobState.setState(JobState.RunningState.SUCCESSFUL);
}
}
@Override
protected void executeCancellation() {
}
}