com.teradata.benchto.driver.execution.BenchmarkExecutionDriver Maven / Gradle / Ivy
/*
* 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 com.teradata.benchto.driver.execution;
import com.google.common.util.concurrent.Futures;
import com.google.common.util.concurrent.ListenableFuture;
import com.google.common.util.concurrent.ListeningExecutorService;
import com.teradata.benchto.driver.Benchmark;
import com.teradata.benchto.driver.BenchmarkExecutionException;
import com.teradata.benchto.driver.Query;
import com.teradata.benchto.driver.concurrent.ExecutorServiceFactory;
import com.teradata.benchto.driver.execution.BenchmarkExecutionResult.BenchmarkExecutionResultBuilder;
import com.teradata.benchto.driver.execution.QueryExecutionResult.QueryExecutionResultBuilder;
import com.teradata.benchto.driver.listeners.benchmark.BenchmarkStatusReporter;
import com.teradata.benchto.driver.macro.MacroService;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.ApplicationContext;
import org.springframework.stereotype.Component;
import javax.sql.DataSource;
import java.sql.Connection;
import java.sql.SQLException;
import java.util.List;
import java.util.concurrent.Callable;
import java.util.concurrent.ExecutionException;
import static com.google.common.base.Preconditions.checkState;
import static com.google.common.collect.Lists.newArrayList;
@Component
public class BenchmarkExecutionDriver
{
private static final Logger LOG = LoggerFactory.getLogger(BenchmarkExecutionDriver.class);
@Autowired
private QueryExecutionDriver queryExecutionDriver;
@Autowired
private BenchmarkStatusReporter statusReporter;
@Autowired
private ExecutorServiceFactory executorServiceFactory;
@Autowired
private MacroService macroService;
@Autowired
private ExecutionSynchronizer executionSynchronizer;
@Autowired
private ApplicationContext applicationContext;
public BenchmarkExecutionResult execute(Benchmark benchmark, int benchmarkOrdinalNumber, int benchmarkTotalCount)
{
LOG.info("[{} of {}] processing benchmark: {}", benchmarkOrdinalNumber, benchmarkTotalCount, benchmark);
BenchmarkExecutionResult benchmarkExecutionResult = null;
try {
macroService.runBenchmarkMacros(benchmark.getBeforeBenchmarkMacros(), benchmark);
benchmarkExecutionResult = executeBenchmark(benchmark);
macroService.runBenchmarkMacros(benchmark.getAfterBenchmarkMacros(), benchmark);
return benchmarkExecutionResult;
}
catch (Exception e) {
if (benchmarkExecutionResult == null || benchmarkExecutionResult.isSuccessful()) {
return failedBenchmarkResult(benchmark, e);
}
else {
checkState(!benchmarkExecutionResult.isSuccessful(), "Benchmark is already failed.");
LOG.error("Error while running after benchmark macros for successful benchmark({})",
benchmark.getAfterBenchmarkMacros(), e);
return benchmarkExecutionResult;
}
}
}
private BenchmarkExecutionResult executeBenchmark(Benchmark benchmark)
{
BenchmarkExecutionResultBuilder resultBuilder = new BenchmarkExecutionResultBuilder(benchmark);
List executions;
try {
executeQueries(benchmark, benchmark.getPrewarmRuns(), false);
executionSynchronizer.awaitAfterBenchmarkExecutionAndBeforeResultReport(benchmark);
statusReporter.reportBenchmarkStarted(benchmark);
resultBuilder = resultBuilder.startTimer();
try {
executions = executeQueries(benchmark, benchmark.getRuns(), true);
}
finally {
resultBuilder = resultBuilder.endTimer();
}
}
catch (RuntimeException e) {
return resultBuilder
.withUnexpectedException(e)
.build();
}
BenchmarkExecutionResult executionResult = resultBuilder
.withExecutions(executions)
.build();
statusReporter.reportBenchmarkFinished(executionResult);
return executionResult;
}
private BenchmarkExecutionResult failedBenchmarkResult(Benchmark benchmark, Exception e)
{
return new BenchmarkExecutionResultBuilder(benchmark)
.withUnexpectedException(e)
.build();
}
@SuppressWarnings("unchecked")
private List executeQueries(Benchmark benchmark, int runs, boolean reportStatus)
{
ListeningExecutorService executorService = executorServiceFactory.create(benchmark.getConcurrency());
try {
List> queryExecutionCallables = buildQueryExecutionCallables(benchmark, runs, reportStatus);
List> executionFutures = (List) executorService.invokeAll(queryExecutionCallables);
return Futures.allAsList(executionFutures).get();
}
catch (InterruptedException | ExecutionException e) {
throw new BenchmarkExecutionException("Could not execute benchmark", e);
}
finally {
executorService.shutdown();
}
}
private List> buildQueryExecutionCallables(Benchmark benchmark, int runs, boolean reportStatus)
{
List> executionCallables = newArrayList();
for (Query query : benchmark.getQueries()) {
for (int run = 1; run <= runs; run++) {
QueryExecution queryExecution = new QueryExecution(benchmark, query, run);
executionCallables.add(() -> {
QueryExecutionResult result;
try (Connection connection = getConnectionFor(queryExecution)) {
macroService.runBenchmarkMacros(benchmark.getBeforeExecutionMacros(), benchmark, connection);
if (reportStatus) {
statusReporter.reportExecutionStarted(queryExecution);
}
try {
result = queryExecutionDriver.execute(queryExecution, connection);
}
catch (Exception e) {
LOG.error("Query Execution failed for benchmark {}", benchmark.getName());
result = new QueryExecutionResultBuilder(queryExecution)
.failed(e)
.build();
}
if (reportStatus) {
statusReporter.reportExecutionFinished(result);
}
macroService.runBenchmarkMacros(benchmark.getAfterExecutionMacros(), benchmark, connection);
}
return result;
});
}
}
return executionCallables;
}
private Connection getConnectionFor(QueryExecution queryExecution)
throws SQLException
{
return applicationContext.getBean(queryExecution.getBenchmark().getDataSource(), DataSource.class).getConnection();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy