org.gradle.cache.internal.AbstractFileLockManagerTest.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 2011 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.cache.internal
import org.apache.commons.lang.RandomStringUtils
import org.gradle.cache.FileIntegrityViolationException
import org.gradle.cache.FileLock
import org.gradle.cache.FileLockManager
import org.gradle.cache.InsufficientLockModeException
import org.gradle.cache.internal.filelock.LockInfoSerializer
import org.gradle.cache.internal.filelock.LockOptionsBuilder
import org.gradle.cache.internal.locklistener.FileLockContentionHandler
import org.gradle.internal.Factory
import org.gradle.internal.concurrent.CompositeStoppable
import org.gradle.internal.id.IdGenerator
import org.gradle.test.fixtures.file.TestFile
import org.gradle.test.fixtures.file.TestNameTestDirectoryProvider
import org.gradle.util.Requires
import org.gradle.util.TestPrecondition
import org.junit.Rule
import spock.lang.Specification
import static org.gradle.cache.FileLockManager.LockMode.Exclusive
import static org.gradle.cache.FileLockManager.LockMode.Shared
abstract class AbstractFileLockManagerTest extends Specification {
@Rule
final TestNameTestDirectoryProvider tmpDir = new TestNameTestDirectoryProvider()
def metaDataProvider = Mock(ProcessMetaDataProvider)
def generator = Stub(IdGenerator)
def contentionHandler = Stub(FileLockContentionHandler)
FileLockManager manager = new DefaultFileLockManager(metaDataProvider, 5000, contentionHandler, generator)
TestFile testFile
TestFile testFileLock
TestFile testDir
TestFile testDirLock
List openedLocks = []
def setup() {
testFile = tmpDir.createFile("state.bin")
testFileLock = tmpDir.file(testFile.name + ".lock")
testDir = tmpDir.createDir("lockable-dir")
testDirLock = tmpDir.file("${testDir.name}/${testDir.name}.lock")
metaDataProvider.processIdentifier >> '123'
metaDataProvider.processDisplayName >> 'process'
contentionHandler.reservePort() >> 34
generator.generateId() >> 678L
}
def cleanup() {
CompositeStoppable.stoppable(openedLocks.toArray()).stop()
}
def "readFile throws integrity exception when not cleanly unlocked file"() {
given:
unlockUncleanly()
and:
def lock = createLock(lockMode)
when:
lock.readFile {}
then:
thrown FileIntegrityViolationException
where:
lockMode << [Exclusive, Shared]
}
def "updateFile throws integrity exception when not cleanly unlocked file"() {
given:
unlockUncleanly()
and:
def lock = createLock(Exclusive)
when:
lock.updateFile {}
then:
thrown FileIntegrityViolationException
}
def "writeFile does not throw integrity exception when not cleanly unlocked file"() {
given:
unlockUncleanly()
when:
createLock(Exclusive).writeFile {}
then:
notThrown FileIntegrityViolationException
}
def "can lock a file"() {
when:
def lock = createLock(lockMode)
then:
lock.isLockFile(tmpDir.createFile(testFile.name + ".lock"))
cleanup:
lock?.close()
where:
lockMode << [Exclusive, Shared]
}
def "can lock a directory"() {
when:
def lock = createLock(lockMode, testDir)
then:
lock.isLockFile(testDirLock)
cleanup:
lock?.close()
where:
lockMode << [Exclusive, Shared]
}
def "can lock a file after it has been closed"() {
given:
def fileLock = createLock(Exclusive)
fileLock.close()
when:
createLock(lockMode)
then:
notThrown(RuntimeException)
where:
lockMode << [Exclusive, Shared]
}
def "lock on new file is not unlocked cleanly"() {
when:
def lock = createLock(mode)
then:
!lock.unlockedCleanly
cleanup:
lock?.close()
where:
mode << [Shared, Exclusive]
}
def "file is 'invalid' unless a valid lock file exists"() {
given:
def lock = createLock(Exclusive)
when:
lock.readFile({})
then:
thrown FileIntegrityViolationException
when:
lock.updateFile({})
then:
thrown FileIntegrityViolationException
when:
lock.writeFile({})
and:
lock.readFile({})
lock.updateFile({})
then:
notThrown FileIntegrityViolationException
}
def "integrity violation exception is thrown after a failed write"() {
given:
def e = new RuntimeException()
def lock = createLock(Exclusive)
when:
lock.writeFile {}
lock.updateFile { throw e }
then:
thrown RuntimeException
when:
lock.readFile({})
then:
thrown FileIntegrityViolationException
}
def "existing lock is unlocked cleanly after writeToFile() has been called"() {
when:
def lock = this.createLock(Exclusive)
lock.writeFile({})
lock.updateFile({} as Runnable)
then:
lock.unlockedCleanly
when:
lock.close()
lock = createLock(Exclusive)
then:
lock.unlockedCleanly
cleanup:
lock?.close()
}
def "existing lock is unlocked cleanly after writeToFile() throws exception"() {
def failure = new RuntimeException()
when:
def lock = createLock(Exclusive)
lock.writeFile({})
lock.updateFile({ throw failure } as Runnable)
then:
RuntimeException e = thrown()
e == failure
!lock.unlockedCleanly
when:
lock.close()
lock = createLock(Exclusive)
then:
!lock.unlockedCleanly
cleanup:
lock?.close()
}
def "cannot lock a file twice in single process"() {
given:
createLock(Exclusive);
when:
createLock(Exclusive);
then:
thrown(IllegalStateException)
}
def "cannot lock twice in single process for mixed modes"() {
given:
createLock(Exclusive);
when:
createLock(Shared);
then:
thrown(IllegalStateException)
}
def "cannot lock twice in single process for shared mode"() {
given:
createLock(Shared);
when:
createLock(Shared);
then:
thrown(IllegalStateException)
}
def "can close a lock multiple times"() {
given:
def lock = createLock(Exclusive)
lock.close()
expect:
lock.close()
}
def "cannot read from file after lock has been closed"() {
given:
def lock = createLock(Exclusive)
lock.close()
when:
lock.readFile({} as Factory)
then:
thrown(IllegalStateException)
}
def "cannot write to file after lock has been closed"() {
given:
def lock = createLock(Exclusive)
lock.close()
when:
lock.updateFile({} as Runnable)
then:
thrown(IllegalStateException)
}
def "can acquire lock on partially written lock file"() {
given:
partiallyWritten()
when:
def lock = createLock(mode)
then:
!lock.unlockedCleanly
lock.isLockFile(testFileLock)
lock.close()
where:
mode << [Shared, Exclusive]
}
def "fails to acquire lock on lock file with unknown version"() {
testFileLock.withDataOutputStream {
it.writeByte(125)
}
when:
createLock(mode)
then:
thrown(IllegalStateException)
where:
mode << [Shared, Exclusive]
}
def "leaves lock file after exclusive lock on new file closed"() {
when:
def lock = createLock(Exclusive)
lock.close()
then:
lock.isLockFile(testFileLock)
and:
isVersionLockFile(testFileLock, true)
}
def "leaves empty lock file after shared lock on new file closed"() {
when:
def lock = createLock(Shared)
lock.close()
then:
lock.isLockFile(testFileLock)
and:
isEmptyLockFile(testFileLock)
when:
lock = createLock(Shared)
lock.close()
then:
lock.isLockFile(testFileLock)
and:
isEmptyLockFile(testFileLock)
}
def "leaves lock file after lock on existing file is closed"() {
given:
writeFile()
when:
def lock = createLock(mode)
lock.close()
then:
lock.isLockFile(testFileLock)
and:
isVersionLockFile(testFileLock, false)
where:
mode << [Shared, Exclusive]
}
@Requires(TestPrecondition.NO_FILE_LOCK_ON_OPEN)
def "writes lock file with info region while exclusive lock is open"() {
expect:
def lock = createLock(Exclusive)
isVersionLockFileWithInfoRegion(testFileLock, true)
lock.writeFile {}
isVersionLockFileWithInfoRegion(testFileLock, false)
cleanup:
lock?.close()
}
@Requires(TestPrecondition.NO_FILE_LOCK_ON_OPEN)
def "writes dirty lock file with info region while updating file"() {
given:
writeFile()
when:
def lock = createLock(Exclusive)
then:
lock.isLockFile(testFileLock)
expect:
isVersionLockFileWithInfoRegion(testFileLock, false)
lock.writeFile {
isVersionLockFileWithInfoRegion(testFileLock, true)
}
isVersionLockFileWithInfoRegion(testFileLock, false)
cleanup:
lock?.close()
}
@Requires(TestPrecondition.NO_FILE_LOCK_ON_OPEN)
def "long descriptor strings are trimmed when written to information region"() {
setup:
def customMetaDataProvider = Mock(ProcessMetaDataProvider)
def processIdentifier = RandomStringUtils.randomAlphanumeric(1000)
1 * customMetaDataProvider.processIdentifier >> processIdentifier
def customManager = new DefaultFileLockManager(customMetaDataProvider, 5000, contentionHandler, generator)
def operationalDisplayName = RandomStringUtils.randomAlphanumeric(1000)
when:
def lock = customManager.lock(testFile, options().withMode(Exclusive), "targetDisplayName", operationalDisplayName)
then:
isVersionLockFileWithInfoRegion(testFileLock, true, processIdentifier.substring(0, LockInfoSerializer.INFORMATION_REGION_DESCR_CHUNK_LIMIT), operationalDisplayName.substring(0, LockInfoSerializer.INFORMATION_REGION_DESCR_CHUNK_LIMIT))
cleanup:
lock?.close()
}
def "require exclusive lock for writing"() {
given:
def lock = createLock(Shared)
when:
lock.writeFile {}
then:
thrown InsufficientLockModeException
}
def "require exclusive lock for updating"() {
given:
writeFile()
def lock = createLock(Shared)
when:
lock.updateFile {}
then:
thrown InsufficientLockModeException
cleanup:
lock?.close()
}
void isEmptyLockFile(TestFile lockFile) {
assert lockFile.isFile()
assert lockFile.length() == 0
}
abstract void isVersionLockFile(TestFile lockFile, boolean dirty)
void isVersionLockFileWithInfoRegion(TestFile lockFile, boolean dirty) {
isVersionLockFileWithInfoRegion(lockFile, dirty, "123", "operation")
}
abstract void isVersionLockFileWithInfoRegion(TestFile lockFile, boolean dirty, String processIdentifier, String operationalName)
FileLock createLock(FileLockManager.LockMode lockMode, File file = testFile, FileLockManager lockManager = manager) {
def lock = lockManager.lock(file, options().withMode(lockMode), "foo", "operation")
openedLocks << lock
lock
}
protected abstract LockOptionsBuilder options();
protected void writeFile(FileLockManager lockManager = manager) {
def lock = lockManager.lock(testFile, options().withMode(Exclusive), "foo", "operation")
try {
lock.writeFile {}
} finally {
lock.close()
}
}
protected void unlockUncleanly(FileLockManager lockManager = manager) {
def lock = createLock(Exclusive, testFile, lockManager)
def failure = new RuntimeException()
try {
lock.writeFile {
throw failure
}
} catch (RuntimeException e) {
if (e != failure) {
throw e
}
} finally {
lock.close()
}
}
protected void partiallyWritten(FileLockManager lockManager = manager) {
createLock(Exclusive, testFile, lockManager).close()
assert testFileLock.length() > 1
def file = new RandomAccessFile(testFileLock, "rw")
try {
file.setLength(1)
} finally {
file.close()
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy