org.gradle.performance.AbstractToolingApiCrossVersionPerformanceTest.groovy Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of gradle-api Show documentation
Show all versions of gradle-api Show documentation
Gradle 6.9.1 API redistribution.
/*
* Copyright 2016 the original author or authors.
*
* 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 org.gradle.performance
import groovy.transform.InheritConstructors
import org.gradle.integtests.fixtures.executer.GradleDistribution
import org.gradle.integtests.fixtures.executer.IntegrationTestBuildContext
import org.gradle.integtests.fixtures.executer.UnderDevelopmentGradleDistribution
import org.gradle.integtests.fixtures.versions.ReleasedVersionDistributions
import org.gradle.integtests.tooling.fixture.ToolingApi
import org.gradle.integtests.tooling.fixture.ToolingApiClasspathProvider
import org.gradle.integtests.tooling.fixture.ToolingApiDistributionResolver
import org.gradle.internal.classloader.ClasspathUtil
import org.gradle.internal.jvm.Jvm
import org.gradle.internal.os.OperatingSystem
import org.gradle.performance.fixture.BuildExperimentSpec
import org.gradle.performance.fixture.CrossVersionPerformanceTestRunner
import org.gradle.performance.fixture.Git
import org.gradle.performance.fixture.InvocationSpec
import org.gradle.performance.fixture.OperationTimer
import org.gradle.performance.fixture.PerformanceTestDirectoryProvider
import org.gradle.performance.fixture.TestProjectLocator
import org.gradle.performance.fixture.TestScenarioSelector
import org.gradle.performance.measure.DataAmount
import org.gradle.performance.measure.Duration
import org.gradle.performance.results.BuildDisplayInfo
import org.gradle.performance.results.CrossVersionPerformanceResults
import org.gradle.performance.results.CrossVersionResultsStore
import org.gradle.performance.results.MeasuredOperationList
import org.gradle.performance.results.ResultsStoreHelper
import org.gradle.test.fixtures.file.TestDirectoryProvider
import org.gradle.test.fixtures.file.TestFile
import org.gradle.test.fixtures.file.TestNameTestDirectoryProvider
import org.gradle.tooling.ProjectConnection
import org.gradle.util.GFileUtils
import org.gradle.util.GradleVersion
import org.junit.Assume
import spock.lang.Specification
abstract class AbstractToolingApiCrossVersionPerformanceTest extends Specification {
protected final static ReleasedVersionDistributions RELEASES = new ReleasedVersionDistributions()
protected final static UnderDevelopmentGradleDistribution CURRENT = new UnderDevelopmentGradleDistribution()
static def resultStore = new CrossVersionResultsStore()
final TestNameTestDirectoryProvider temporaryFolder = new PerformanceTestDirectoryProvider()
protected ToolingApiExperimentSpec experimentSpec
// caching class loaders at this level because for performance experiments
// we don't want caches of the TAPI to be visible between different experiments
protected final Map testClassLoaders = [:]
protected ClassLoader tapiClassLoader
public Class tapiClass(Class clazz) {
tapiClassLoader.loadClass(clazz.name)
}
void experiment(String projectName, String displayName, @DelegatesTo(ToolingApiExperimentSpec) Closure> spec) {
experimentSpec = new ToolingApiExperimentSpec(displayName, projectName, temporaryFolder.testDirectory, 3, 10, 5000L, 500L, null)
def clone = spec.rehydrate(experimentSpec, this, this)
clone.resolveStrategy = Closure.DELEGATE_FIRST
clone.call(experimentSpec)
}
CrossVersionPerformanceResults performMeasurements() {
new Measurement().run()
}
static {
// TODO - find a better way to cleanup
System.addShutdownHook {
((Closeable) resultStore).close()
}
}
@InheritConstructors
public static class ToolingApiExperimentSpec extends BuildExperimentSpec {
List targetVersions = []
List extraTestClassPath = []
Closure> action
void action(@DelegatesTo(ProjectConnection) Closure> action) {
this.action = action
}
@Override
BuildDisplayInfo getDisplayInfo() {
new BuildDisplayInfo(projectName, displayName, [], [], [], true)
}
@Override
InvocationSpec getInvocation() {
throw new UnsupportedOperationException('Invocations are not supported for Tooling API performance tests')
}
}
private class Measurement implements ToolingApiClasspathProvider {
private CrossVersionPerformanceResults run() {
def testId = experimentSpec.displayName
def scenarioSelector = new TestScenarioSelector()
Assume.assumeTrue(scenarioSelector.shouldRun(testId, [experimentSpec.projectName].toSet(), resultStore))
def testProjectLocator = new TestProjectLocator()
def projectDir = testProjectLocator.findProjectDir(experimentSpec.projectName)
IntegrationTestBuildContext buildContext = new IntegrationTestBuildContext()
def results = new CrossVersionPerformanceResults(
testId: testId,
previousTestIds: [],
testProject: experimentSpec.projectName,
jvm: Jvm.current().toString(),
operatingSystem: OperatingSystem.current().toString(),
versionUnderTest: GradleVersion.current().getVersion(),
vcsBranch: Git.current().branchName,
vcsCommits: [Git.current().commitId],
startTime: System.currentTimeMillis(),
tasks: [],
args: [],
gradleOpts: [],
daemon: true,
channel: ResultsStoreHelper.determineChannel())
def resolver = new ToolingApiDistributionResolver().withDefaultRepository()
try {
List baselines = CrossVersionPerformanceTestRunner.toBaselineVersions(RELEASES, experimentSpec.targetVersions).toList()
[*baselines, 'current'].each { String version ->
def workingDirProvider = copyTemplateTo(projectDir, experimentSpec.workingDirectory, version)
GradleDistribution dist = 'current' == version ? CURRENT : buildContext.distribution(version)
println "Testing ${dist.version}..."
if ('current' != version) {
def baselineVersion = results.baseline(version)
}
def toolingApiDistribution = resolver.resolve(dist.version.version)
def testClassPath = [*experimentSpec.extraTestClassPath]
// add TAPI test fixtures to classpath
testClassPath << ClasspathUtil.getClasspathForClass(ToolingApi)
tapiClassLoader = getTestClassLoader(testClassLoaders, toolingApiDistribution, testClassPath) {
}
def tapiClazz = tapiClassLoader.loadClass(ToolingApi.name)
def toolingApi = tapiClazz.newInstance(dist, workingDirProvider)
assert toolingApi != ToolingApi
warmup(toolingApi)
println "Waiting ${experimentSpec.sleepAfterWarmUpMillis}ms before measurements"
sleep(experimentSpec.sleepAfterWarmUpMillis)
measure(results, toolingApi, version)
}
} finally {
resolver.stop()
}
results.endTime = System.currentTimeMillis();
results.assertEveryBuildSucceeds()
resultStore.report(results)
results
}
private TestDirectoryProvider copyTemplateTo(File templateDir, File workingDir, String version) {
TestFile perVersionDir = new TestFile(workingDir, version)
if (!perVersionDir.exists()) {
perVersionDir.mkdirs()
} else {
throw new IllegalArgumentException("Didn't expect to find an existing directory at $perVersionDir")
}
GFileUtils.copyDirectory(templateDir, perVersionDir)
return new TestDirectoryProvider() {
@Override
TestFile getTestDirectory() {
perVersionDir
}
@Override
void suppressCleanup() {
}
}
}
private void measure(CrossVersionPerformanceResults results, toolingApi, String version) {
OperationTimer timer = new OperationTimer()
MeasuredOperationList versionResults = 'current' == version ? results.current : results.version(version).results
experimentSpec.with {
invocationCount.times { n ->
println "Run #${n + 1}"
def measuredOperation = timer.measure {
toolingApi.withConnection(action)
}
measuredOperation.configurationTime = Duration.millis(0)
measuredOperation.executionTime = Duration.millis(0)
// TODO: cc find a way to collect memory stats
measuredOperation.maxCommittedHeap = DataAmount.mbytes(0)
measuredOperation.maxHeapUsage = DataAmount.mbytes(0)
measuredOperation.maxUncollectedHeap = DataAmount.mbytes(0)
measuredOperation.totalHeapUsage = DataAmount.mbytes(0)
measuredOperation.totalMemoryUsed = DataAmount.mbytes(0)
versionResults.add(measuredOperation)
sleep(sleepAfterTestRoundMillis)
}
}
}
private void warmup(toolingApi) {
experimentSpec.with {
warmUpCount.times { n ->
println "Warm-up #${n + 1}"
toolingApi.withConnection(action)
sleep(sleepAfterTestRoundMillis)
}
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy