org.gradle.caching.internal.controller.DefaultBuildCacheControllerPackOperationExecutorTest.groovy Maven / Gradle / Ivy
/*
* Copyright 2022 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.caching.internal.controller
import groovy.transform.Immutable
import org.gradle.api.internal.cache.StringInterner
import org.gradle.caching.BuildCacheKey
import org.gradle.caching.internal.CacheableEntity
import org.gradle.caching.internal.controller.operations.PackOperationResult
import org.gradle.caching.internal.controller.operations.UnpackOperationResult
import org.gradle.caching.internal.origin.OriginMetadata
import org.gradle.caching.internal.origin.OriginMetadataFactory
import org.gradle.caching.internal.origin.OriginReader
import org.gradle.caching.internal.origin.OriginWriter
import org.gradle.caching.internal.packaging.BuildCacheEntryPacker
import org.gradle.internal.file.FileMetadata
import org.gradle.internal.file.TreeType
import org.gradle.internal.file.impl.DefaultFileMetadata
import org.gradle.internal.hash.TestHashCodes
import org.gradle.internal.operations.BuildOperationContext
import org.gradle.internal.operations.BuildOperationExecutor
import org.gradle.internal.operations.CallableBuildOperation
import org.gradle.internal.operations.RunnableBuildOperation
import org.gradle.internal.snapshot.DirectorySnapshot
import org.gradle.internal.snapshot.FileSystemLocationSnapshot
import org.gradle.internal.snapshot.RegularFileSnapshot
import org.gradle.internal.snapshot.SnapshotVisitorUtil
import org.gradle.internal.vfs.FileSystemAccess
import org.gradle.test.fixtures.file.TestNameTestDirectoryProvider
import org.junit.Rule
import spock.lang.Specification
import java.time.Duration
import static org.gradle.caching.internal.controller.DefaultBuildCacheController.PackOperationExecutor
import static org.gradle.internal.file.TreeType.DIRECTORY
import static org.gradle.internal.file.TreeType.FILE
class DefaultBuildCacheControllerPackOperationExecutorTest extends Specification {
@Rule
TestNameTestDirectoryProvider temporaryFolder = new TestNameTestDirectoryProvider(getClass())
def packer = Mock(BuildCacheEntryPacker)
def originFactory = Mock(OriginMetadataFactory)
def fileSystemAccess = Mock(FileSystemAccess)
def stringInterner = new StringInterner()
def buildOperationContext = Mock(BuildOperationContext)
def buildOperationExecutor = Mock(BuildOperationExecutor)
PackOperationExecutor packOperationExecutor = new PackOperationExecutor(buildOperationExecutor, fileSystemAccess, packer, originFactory, stringInterner)
def key = Mock(BuildCacheKey)
def originMetadata = Mock(OriginMetadata)
def originReader = Mock(OriginReader)
def originWriter = Mock(OriginWriter)
def "unpack invokes unpacker and fingerprints trees"() {
def outputFile = temporaryFolder.file("output.txt")
def outputDir = temporaryFolder.file("outputDir")
def outputDirFile = outputDir.file("file.txt")
def input = temporaryFolder.createFile("input")
def entity = entity(
prop("outputDir", DIRECTORY, outputDir),
prop("outputFile", FILE, outputFile)
)
def outputFileSnapshot = new RegularFileSnapshot(outputFile.absolutePath, outputFile.name, TestHashCodes.hashCodeFrom(234), DefaultFileMetadata.file(15, 234, FileMetadata.AccessType.DIRECT))
def fileSnapshots = [
outputDir: new DirectorySnapshot(outputDir.getAbsolutePath(), outputDir.name, FileMetadata.AccessType.DIRECT, TestHashCodes.hashCodeFrom(456), [
new RegularFileSnapshot(outputDirFile.getAbsolutePath(), outputDirFile.name, TestHashCodes.hashCodeFrom(123), DefaultFileMetadata.file(46, 123, FileMetadata.AccessType.DIRECT))
]),
outputFile: outputFileSnapshot
]
when:
def result = packOperationExecutor.unpack(key, entity, input)
then:
1 * buildOperationExecutor.call(_) >> { CallableBuildOperation action -> action.call(buildOperationContext)}
1 * originFactory.createReader(entity) >> originReader
1 * fileSystemAccess.write([outputDir.absolutePath, outputFile.absolutePath], _)
then:
1 * packer.unpack(entity, _ as InputStream, originReader) >> new BuildCacheEntryPacker.UnpackResult(originMetadata, 123L, fileSnapshots)
then:
1 * fileSystemAccess.record(_ as DirectorySnapshot) >> { FileSystemLocationSnapshot snapshot ->
assert snapshot.absolutePath == outputDir.absolutePath
assert snapshot.name == outputDir.name
}
1 * fileSystemAccess.record(_ as RegularFileSnapshot) >> { FileSystemLocationSnapshot snapshot ->
assert snapshot.absolutePath == outputFileSnapshot.absolutePath
assert snapshot.name == outputFileSnapshot.name
assert snapshot.hash == outputFileSnapshot.hash
}
then:
1 * buildOperationContext.setResult(_) >> { args ->
assert (args[0] as UnpackOperationResult).archiveEntryCount == 123
}
then:
result.artifactEntryCount == 123
result.originMetadata == originMetadata
result.resultingSnapshots.keySet() as List == ["outputDir", "outputFile"]
SnapshotVisitorUtil.getAbsolutePaths(result.resultingSnapshots["outputFile"], true) == [outputFile.absolutePath]
SnapshotVisitorUtil.getAbsolutePaths(result.resultingSnapshots["outputDir"], true) == [outputDir.absolutePath, outputDirFile.absolutePath]
0 * _
}
def "after failed unpacking error is propagated and output is not removed"() {
def input = temporaryFolder.createFile("input")
def outputFile = temporaryFolder.file("output.txt")
def entity = this.entity(prop("output", FILE, outputFile))
when:
packOperationExecutor.unpack(key, entity, input)
then:
1 * buildOperationExecutor.call(_) >> { CallableBuildOperation action -> action.call(buildOperationContext)}
1 * originFactory.createReader(entity) >> originReader
1 * fileSystemAccess.write([outputFile.absolutePath], _)
then:
1 * packer.unpack(entity, _ as InputStream, originReader) >> {
outputFile << "partially extracted output fil..."
throw new RuntimeException("unpacking error")
}
then:
def ex = thrown Exception
ex.message == "unpacking error"
outputFile.exists()
0 * _
}
def "pack invokes packer"() {
def output = temporaryFolder.createFile("output")
def entity = entity(prop("output"))
def outputSnapshots = Mock(Map)
when:
packOperationExecutor.pack(output, key, entity, outputSnapshots, Duration.ofMillis(421L))
then:
1 * buildOperationExecutor.run(_) >> { RunnableBuildOperation action -> action.run(buildOperationContext)}
1 * originFactory.createWriter(entity, Duration.ofMillis(421L)) >> originWriter
then:
1 * packer.pack(entity, outputSnapshots, _ as OutputStream, originWriter) >> new BuildCacheEntryPacker.PackResult(123)
then:
1 * buildOperationContext.setResult(_) >> { args ->
def packResult = args[0] as PackOperationResult
assert packResult.archiveEntryCount == 123
assert packResult.archiveSize == output.size()
}
then:
0 * _
}
def entity(TestCacheableTree... trees) {
return Stub(CacheableEntity) {
visitOutputTrees(_ as CacheableEntity.CacheableTreeVisitor) >> { CacheableEntity.CacheableTreeVisitor visitor ->
trees.each { visitor.visitOutputTree(it.name, it.type, it.root) }
}
}
}
TestCacheableTree prop(String name, TreeType type = FILE, File root = null) {
new TestCacheableTree(name, type, root)
}
@Immutable(knownImmutableClasses = [File])
private static class TestCacheableTree {
String name
TreeType type
File root
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy