org.gradle.api.CrossBuildScriptCachingIntegrationSpec.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.api
import org.gradle.integtests.fixtures.AbstractIntegrationSpec
import org.gradle.integtests.fixtures.daemon.DaemonLogsAnalyzer
import org.gradle.integtests.fixtures.daemon.DaemonsFixture
import org.gradle.test.fixtures.file.TestFile
import org.gradle.test.fixtures.server.http.BlockingHttpServer
import org.gradle.util.GradleVersion
import org.junit.Rule
import spock.lang.Issue
class CrossBuildScriptCachingIntegrationSpec extends AbstractIntegrationSpec {
FileTreeBuilder root
File cachesDir
File scriptCachesDir
File remappedCachesDir
@Rule
BlockingHttpServer server = new BlockingHttpServer()
private TestFile homeDirectory = testDirectoryProvider.getTestDirectory().file("user-home")
def setup() {
executer.requireOwnGradleUserHomeDir()
root = new FileTreeBuilder(testDirectory)
cachesDir = new File(homeDirectory, 'caches')
def versionCaches = new File(cachesDir, GradleVersion.current().version)
scriptCachesDir = new File(versionCaches, 'scripts')
remappedCachesDir = new File(versionCaches, 'scripts-remapped')
}
def "identical build files are compiled once"() {
given:
root {
core {
'core.gradle'(this.simpleBuild())
}
module1 {
'module1.gradle'(this.simpleBuild())
}
'settings.gradle'(settings('core', 'module1'))
}
when:
run 'help'
then:
def settingsHash = uniqueRemapped('settings')
def coreHash = uniqueRemapped('core')
def module1Hash = uniqueRemapped('module1')
and:
remappedCacheSize() == 3 // one for each build script
scriptCacheSize() == 2 // one for settings, one for the 2 identical scripts
coreHash == module1Hash
hasCachedScripts(settingsHash, coreHash)
}
def "identical build files are compiled once for distinct invocations"() {
given:
root {
core {
'core.gradle'(this.simpleBuild())
}
module1 {
'module1.gradle'(this.simpleBuild())
}
'settings.gradle'(settings('core', 'module1'))
}
when:
executer.withGradleUserHomeDir(homeDirectory)
executer.requireDaemon()
executer.requireIsolatedDaemons()
run 'help'
run 'help'
then:
def settingsHash = uniqueRemapped('settings')
def coreHash = uniqueRemapped('core')
def module1Hash = uniqueRemapped('module1')
and:
remappedCacheSize() == 3 // one for each build script
scriptCacheSize() == 2 // one for settings, one for the 2 identical scripts
coreHash == module1Hash
hasCachedScripts(settingsHash, coreHash)
getCompileClasspath(coreHash, 'proj').length == 1
}
def "can have two build files with same contents and file name"() {
given:
root {
core {
'build.gradle'(this.simpleBuild())
}
module1 {
'build.gradle'(this.simpleBuild())
}
'settings.gradle'("include 'core', 'module1'")
}
when:
run 'help'
then:
def settingsHash = uniqueRemapped('settings')
def buildHashes = hasRemapped('build')
and:
remappedCacheSize() == 3 // one for each build script
scriptCacheSize() == 2 // one for settings, one for the 2 identical scripts
buildHashes.size() == 2 // two build.gradle files in different dirs
hasCachedScripts(settingsHash, *buildHashes)
}
def "can have two build files with different contents and same file name"() {
given:
root {
core {
'build.gradle'(this.simpleBuild())
}
module1 {
'build.gradle'(this.simpleBuild('different contents'))
}
'settings.gradle'("include 'core', 'module1'")
}
when:
run 'help'
then:
def settingsHash = uniqueRemapped('settings')
def buildHashes = hasRemapped('build')
and:
remappedCacheSize() == 3 // one for each build script
scriptCacheSize() == 3 // one for settings, one for each build.gradle file
buildHashes.size() == 2 // two build.gradle files in different dirs
hasCachedScripts(settingsHash, *buildHashes)
}
def "cache size increases when build file changes"() {
given:
root {
core {
'build.gradle'(this.simpleBuild())
}
module1 {
'build.gradle'(this.simpleBuild())
}
'settings.gradle'("include 'core', 'module1'")
}
run 'help'
when:
file('module1/build.gradle').text = simpleBuild('different contents')
run 'help'
then:
def settingsHash = uniqueRemapped('settings')
def buildHashes = hasRemapped('build')
remappedCacheSize() == 3 // one for each build script
scriptCacheSize() == 3 // one for settings, one for each build.gradle file
buildHashes.size() == 3 // two build.gradle files in different dirs + one new build.gradle
hasCachedScripts(settingsHash, *buildHashes)
}
def "remapping scripts doesn't mix up classes with same name"() {
given:
root {
'build.gradle'('''
task greet()
apply from: "gradle/one.gradle"
apply from: "gradle/two.gradle"
''')
gradle {
'one.gradle'('''
class Greeter { String toString() { 'Greetings from One!' } }
greet.doLast() { println new Greeter() }
''')
'two.gradle'('''
class Greeter { String toString() { 'Greetings from Two!' } }
greet.doLast() { println new Greeter() }
''')
}
}
when:
run 'greet'
then:
outputContains 'Greetings from One!'
outputContains 'Greetings from Two!'
}
def "reports errors at the correct location when 2 scripts are identical"() {
given:
root {
module1 {
'module1.gradle'(this.taskThrowingError())
}
module2 {
'module2.gradle'(this.taskThrowingError())
}
'settings.gradle'(settings('module1', 'module2'))
}
when:
fails 'module1:someTask'
then:
def settingsHash = uniqueRemapped('settings')
def module1Hash = uniqueRemapped('module1')
def module2Hash = uniqueRemapped('module2')
and:
remappedCacheSize() == 3 // one for each build script
scriptCacheSize() == 2 // one for settings, one for the 2 identical scripts
module1Hash == module2Hash
hasCachedScripts(settingsHash, module1Hash)
and:
def module1File = file("module1/module1.gradle")
failure.assertHasFileName("Build file '$module1File'")
failure.assertHasLineNumber(4)
when:
fails 'module2:someTask'
then:
def module2File = file("module2/module2.gradle")
failure.assertHasFileName("Build file '$module2File'")
failure.assertHasLineNumber(4)
}
def "caches scripts applied from remote locations"() {
server.start()
given:
root {
'build.gradle'(this.applyFromRemote(server))
}
server.expect(server.get("shared.gradle").send("println 'Echo'"))
when:
run 'tasks'
then:
outputContains 'Echo'
def buildHash = uniqueRemapped('build')
def sharedHash = uniqueRemapped('shared')
and:
remappedCacheSize() == 2 // one for each build script
scriptCacheSize() == 2 // one for each build script
hasCachedScripts(buildHash, sharedHash)
}
def "caches scripts applied from remote locations when remote script changes"() {
server.start()
given:
root {
'build.gradle'(this.applyFromRemote(server))
}
def buildHash
def sharedHash
server.expect(server.get("shared.gradle").send("println 'Echo 0'"))
when:
run 'tasks'
buildHash = uniqueRemapped('build')
sharedHash = uniqueRemapped('shared')
then:
outputContains 'Echo 0'
and:
remappedCacheSize() == 2 // one for each build script
scriptCacheSize() == 2 // one for each build script
hasCachedScripts(buildHash, sharedHash)
when:
server.expect(server.head("shared.gradle"))
server.expect(server.get("shared.gradle").send("println 'Echo 1'"))
run 'tasks'
buildHash = uniqueRemapped('build')
def sharedHashs = hasRemapped('shared')
then:
outputContains 'Echo 1'
and:
remappedCacheSize() == 2 // one for each build script
scriptCacheSize() == 3 // one for each build script of this invocation + 1 from the previous invocation
hasCachedScripts(buildHash, *sharedHashs)
}
@Issue("GRADLE-2795")
def "can change script while build is running"() {
server.start()
given:
buildFile << """
task someLongRunningTask {
doLast {
${server.callFromBuild("running")}
}
}
"""
def handle = server.expectAndBlock("running")
when:
def longRunning = executer.withTasks("someLongRunningTask").start()
handle.waitForAllPendingCalls()
then:
remappedCacheSize() == 1 // build.gradle
scriptCacheSize() == 1 // build.gradle
hasCachedScripts(uniqueRemapped('build'))
when:
buildFile << """
task fastTask { }
"""
executer.withTasks("fastTask").run()
assert longRunning.isRunning()
handle.releaseAll()
longRunning.waitForExit()
then:
remappedCacheSize() == 1 // build.gradle
scriptCacheSize() == 2 // build.gradle version 1, build.gradle version 2
hasCachedScripts(*hasRemapped('build'))
}
def "build script is recompiled when project's classpath changes"() {
createJarWithProperties("lib/foo.jar", [source: 1])
root {
'build.gradle'('''
buildscript {
dependencies {
classpath files('lib/foo.jar')
}
}
''')
}
when:
run 'help'
then:
def coreHash = uniqueRemapped('build')
remappedCacheSize() == 1
scriptCacheSize() == 1
hasCachedScripts(coreHash)
getCompileClasspath(coreHash, 'proj').length == 1
when:
sleep(1000)
createJarWithProperties("lib/foo.jar", [target: 2])
run 'help'
coreHash = uniqueRemapped('build')
then:
remappedCacheSize() == 1
scriptCacheSize() == 1
hasCachedScripts(coreHash)
getCompileClasspath(coreHash, 'proj').length == 2
}
def "build script is recompiled when parent project's classpath changes"() {
createJarWithProperties("lib/foo.jar", [source: 1])
root {
'build.gradle'('''
buildscript {
dependencies {
classpath files('lib/foo.jar')
}
}
''')
module {
'module.gradle'(this.simpleBuild('module'))
}
'settings.gradle'(this.settings('module'))
}
when:
run 'help'
then:
def coreHash = uniqueRemapped('build')
def moduleHash = uniqueRemapped('module')
def settingsHash = uniqueRemapped('settings')
remappedCacheSize() == 3 // core, module, settings
scriptCacheSize() == 3
hasCachedScripts(coreHash, moduleHash, settingsHash)
getCompileClasspath(coreHash, 'proj').length == 1
getCompileClasspath(moduleHash, 'proj').length == 1
when:
sleep(1000)
createJarWithProperties("lib/foo.jar", [target: 2])
run 'help'
coreHash = uniqueRemapped('build')
moduleHash = uniqueRemapped('module')
settingsHash = uniqueRemapped('settings')
then:
remappedCacheSize() == 3
scriptCacheSize() == 3
hasCachedScripts(coreHash, moduleHash, settingsHash)
getCompileClasspath(coreHash, 'proj').length == 2
getCompileClasspath(moduleHash, 'proj').length == 2
}
def "init script is cached"() {
root {
'build.gradle'(this.simpleBuild())
gradle {
'init.gradle'('// init script')
}
}
when:
executer.withArgument('-Igradle/init.gradle')
run 'help'
then:
def initHash = uniqueRemapped('init')
def coreHash = uniqueRemapped('build')
remappedCacheSize() == 2
scriptCacheSize() == 2
hasCachedScripts(coreHash, initHash)
getCompileClasspath(coreHash, 'proj').length == 1
getCompileClasspath(initHash, 'init').length == 1
}
def "same script can be applied from init script, settings script and build script"() {
root {
'common.gradle'('println "poke"')
'init.gradle'('''
// init script
apply from: 'common.gradle'
''')
'settings.gradle'('''
// settings script
apply from: 'common.gradle'
''')
'build.gradle'('''
// build script
apply from: 'common.gradle'
''')
}
when:
executer.withArgument('-Iinit.gradle')
run 'help'
then:
def commonHash = uniqueRemapped('common')
def initHash = uniqueRemapped('settings')
def settingsHash = uniqueRemapped('init')
def coreHash = uniqueRemapped('build')
remappedCacheSize() == 4
scriptCacheSize() == 4
hasCachedScripts(commonHash, settingsHash, coreHash, initHash)
getCompileClasspath(commonHash, 'cp_dsl').length == 2 // settings classpath is deprecated so it differs
getCompileClasspath(commonHash, 'dsl').length == 2 // settings classpath is deprecated so it differs
}
def "same script can be applied from identical init script, settings script and build script"() {
root {
'common.gradle'('println "poke"')
'init.gradle'('''
apply from: 'common.gradle'
''')
'settings.gradle'('''
apply from: 'common.gradle'
''')
'build.gradle'('''
apply from: 'common.gradle'
''')
}
when:
executer.withArgument('-Iinit.gradle')
run 'help'
then:
def commonHash = uniqueRemapped('common')
def initHash = uniqueRemapped('settings')
def settingsHash = uniqueRemapped('init')
def coreHash = uniqueRemapped('build')
remappedCacheSize() == 4
scriptCacheSize() == 2
hasCachedScripts(commonHash, settingsHash, coreHash, initHash)
getCompileClasspath(commonHash, 'cp_dsl').length == 2
getCompileClasspath(commonHash, 'dsl').length == 2
}
def "remapped classes have script origin"() {
root {
'build.gradle'('''
void assertScriptOrigin(Object o, Set seen) {
assert (o instanceof org.gradle.internal.scripts.ScriptOrigin)
// need to get through reflection to bypass the Groovy MOP on closures, which would cause calling the method on the owner instead of the closure itself
def originalClassName = o.class.getMethod('getOriginalClassName').invoke(o)
def contentHash = o.class.getMethod('getContentHash').invoke(o)
assert originalClassName
assert contentHash
println "Action type: ${originalClassName} (remapped name: ${o.class})"
println "Action hash: ${contentHash}"
if (!seen.add(contentHash)) {
throw new AssertionError("Expected a unique hash, but found duplicate: ${o.contentHash} in $seen")
}
}
Set seen = []
assertScriptOrigin(this, seen)
task one {
doLast {
{ ->
assertScriptOrigin(owner, seen) // hack to get a handle on the parent closure
}()
}
}
task two {
def v
v = { assertScriptOrigin(v, seen) }
doFirst(v)
}
task three {
doLast(new Action() {
void execute(Object o) {
assertScriptOrigin(this, seen)
}
})
}
task four {
doLast {
def a = new A()
assertScriptOrigin(a, seen)
}
}
class A {}
''')
}
when:
run 'one', 'two', 'three', 'four'
then:
noExceptionThrown()
}
def "same applied script is compiled once for different projects with different classpath"() {
root {
'common.gradle'('println "poke"')
}
when:
def iterations = 3
def builder = root
iterations.times { n ->
createJarWithProperties("foo${n}.jar", [value: n])
new File(root.baseDir, 'build.gradle').delete()
builder {
'build.gradle'("""
buildscript {
dependencies {
classpath files('foo${n}.jar')
}
}
apply from: 'common.gradle'
""")
}
run 'help'
sleep(1000)
}
then:
remappedCacheSize() == 2 // build + common
scriptCacheSize() == 1 + iterations // common + 1 build script per iteration
}
def "script don't get recompiled if daemon disappears"() {
root {
buildSrc {
'build.gradle'('''
apply plugin: 'java'
''')
src {
main {
java {
'Foo.java'('public class Foo {}')
}
}
}
}
'build.gradle'('''apply from:'main.gradle' ''')
'main.gradle'('''
task success {
doLast {
println 'ok'
}
}
''')
}
executer.requireIsolatedDaemons()
executer.requireDaemon()
executer.withGradleUserHomeDir(homeDirectory)
when:
succeeds 'success'
daemons.daemon.kill()
succeeds 'success'
then:
String hash = uniqueRemapped('main')
getCompileClasspath(hash, 'dsl').length == 1
}
DaemonsFixture getDaemons() {
new DaemonLogsAnalyzer(executer.daemonBaseDir)
}
int buildScopeCacheSize() {
def m = output =~ /(?s).*Build scope cache size: (\d+).*/
m.matches()
m.group(1).toInteger()
}
int crossBuildScopeCacheSize() {
def m = output =~ /(?s).*Cross-build scope cache size: (\d+).*/
m.matches()
m.group(1).toInteger()
}
Set buildScopeCacheContents() {
def m = output =~ /(?s).*Build scope cache contents: \[(.+?)\].*/
m.matches()
m.group(1).split(", ") as Set
}
Set crossBuildScopeCacheContents() {
def m = output =~ /(?s).*Cross-build scope cache contents: \[(.+?)\].*/
m.matches()
m.group(1).split(", ") as Set
}
Map crossBuildCacheStats() {
def m = output =~ /(?s).*Cross-build scope cache stats: CacheStats\{((?:(?:\p{Alnum}+=\d+)(?:, )?)+)}.*/
m.matches()
def stats = [:]
m = m.group(1) =~ /(\p{Alnum}+)=(\d+)/
while (m.find()) {
stats[m.group(1)] = m.group(2).toInteger()
}
stats
}
List hasRemapped(String buildFile) {
def remapped = remappedCachesDir.listFiles().findAll { it.name.startsWith(buildFile) }
if (remapped) {
def contentHash = remapped*.list().flatten()
return contentHash
}
throw new AssertionError("Cannot find a remapped build script for '${buildFile}.gradle'")
}
String uniqueRemapped(String buildFile) {
def hashes = hasRemapped(buildFile)
assert hashes.size() == 1
hashes[0]
}
int remappedCacheSize() {
remappedCachesDir.list().length
}
int scriptCacheSize() {
scriptCachesDir.list().length
}
void hasCachedScripts(String... contentHashes) {
Set foundInCache = scriptCachesDir.list() as Set
Set expected = contentHashes as Set
assert foundInCache == expected
}
String[] getCompileClasspath(String contentHash, String dslId) {
new File(new File(scriptCachesDir, contentHash), dslId).list()?:new String[0]
}
void hasCachedScriptForClasspath(String contentHash, String dslId, String classpathHash) {
def contentsDir = new File(scriptCachesDir, contentHash)
assert contentsDir.exists(): "Unable to find a cached script directory for content hash $contentHash. Found: ${scriptCachesDir.list().toList()}"
def dslDir = new File(contentsDir, dslId)
assert dslDir.exists(): "Unable to find a cached script directory for content hash $contentHash and DSL id $dslId. Found: ${contentsDir.list().toList()}"
def classpathDir = new File(dslDir, classpathHash)
assert classpathDir.exists(): "Unable to find a cached script directory for content hash $contentHash, DSL id $dslId and classpath hash $classpathHash. Found: ${dslDir.list().toList()}"
}
String simpleBuild(String comment = '') {
"""
// ${comment}
apply plugin:'java'
"""
}
String settings(String... projects) {
String includes = "include ${projects.collect { "'$it'" }.join(', ')}"
"""
$includes
rootProject.children.each { project ->
project.projectDir = new File(project.name)
project.buildFileName = "\${project.name}.gradle"
}
"""
}
String taskThrowingError() {
'''
task someTask() {
doLast {
thisMethodDoesNotExist()
}
}
'''
}
String applyFromRemote(BlockingHttpServer server) {
"""
apply from: '${server.uri}/shared.gradle'
"""
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy