Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Exclusive Cache Lock #8209

Open
wants to merge 42 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
42 commits
Select commit Hold shift + click to select a range
8674a8b
Cache Lock
yschimke Jan 20, 2024
c3df60f
Merge branch 'master' into cache_singleton
yschimke Apr 1, 2024
166a255
Merge branch 'master' into cache_singleton
yschimke Apr 1, 2024
b02738d
Rework
yschimke Apr 1, 2024
ac0cc79
Rework
yschimke Apr 1, 2024
2525edf
Add test
yschimke Apr 1, 2024
7072358
Cleanup
yschimke Apr 1, 2024
5ca131f
Activate missed test
yschimke Apr 1, 2024
0a2fbb6
Cleanup
yschimke Apr 1, 2024
c9db9d4
Avoid checking on non system file systems
yschimke Apr 1, 2024
97e566b
Revert "Avoid checking on non system file systems"
yschimke Apr 1, 2024
ee1daa6
Avoid failing on dns tests
yschimke Apr 1, 2024
3041258
Adding a test
yschimke Apr 2, 2024
7faba3e
Fixes
yschimke Apr 2, 2024
44d3640
Test on more platforms
yschimke Apr 3, 2024
792b689
Cleanup
yschimke Apr 3, 2024
5117f46
Merge branch 'master' into cache_singleton
yschimke Apr 3, 2024
a922b1d
Cleanup
yschimke Apr 3, 2024
e24e146
Fixes
yschimke Apr 6, 2024
04c057f
Fixes
yschimke Apr 6, 2024
5c6c60d
Cache fixes
yschimke Apr 6, 2024
12b7eb9
Cache fixes
yschimke Apr 6, 2024
ae3830d
Cache fixes
yschimke Apr 6, 2024
99a5ee5
Cache fixes
yschimke Apr 6, 2024
cc97769
Cache fixes
yschimke Apr 6, 2024
6dbb03f
Fix test
yschimke Apr 6, 2024
19be22b
Merge branch 'refs/heads/master' into cache_singleton
yschimke Apr 6, 2024
feafff9
Fix test
yschimke Apr 6, 2024
72359a6
avoid windows for now
yschimke Apr 6, 2024
b76aa7f
Fix or skip on windows
yschimke Apr 7, 2024
89e9d80
cleanup
yschimke Apr 7, 2024
28025e2
skip on windows
yschimke Apr 7, 2024
fa707bc
Update DuplexTest.kt
yschimke Apr 7, 2024
c5a87f0
Merge branch 'master' into cache_singleton
yschimke Jan 4, 2025
bb0d4b0
Fixes
yschimke Jan 4, 2025
192e93b
Fixes
yschimke Jan 4, 2025
b4115b2
Fixes
yschimke Jan 4, 2025
2f959af
Fixes
yschimke Jan 4, 2025
7af5705
Fixes
yschimke Jan 4, 2025
4e9d4b3
Fix test
yschimke Jan 11, 2025
019560c
reformat
yschimke Jan 11, 2025
433f6f7
Make it optional
yschimke Jan 11, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -865,6 +865,7 @@ class OkHttpTest {
assertEquals(setOf(OkHttpTest::class.java.name), testHandler.calls.keys)
}

@Test
fun testCachedRequest() {
enableTls()

Expand Down
9 changes: 9 additions & 0 deletions okcurl/src/test/kotlin/okhttp3/curl/MainTest.kt
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,9 @@ import com.github.ajalt.clikt.core.parse
import java.io.IOException
import kotlin.test.Test
import okhttp3.RequestBody
import okhttp3.TestUtil
import okio.Buffer
import org.junit.jupiter.api.BeforeAll

class MainTest {
@Test
Expand Down Expand Up @@ -138,5 +140,12 @@ class MainTest {
throw RuntimeException(e)
}
}

@JvmStatic
@BeforeAll
fun beforeAll() {
// https://github.com/square/okhttp/issues/8342
TestUtil.assumeNotWindows()
}
}
}
11 changes: 11 additions & 0 deletions okcurl/src/test/kotlin/okhttp3/curl/OkcurlTest.kt
Original file line number Diff line number Diff line change
Expand Up @@ -17,10 +17,21 @@ package okhttp3.curl

import com.github.ajalt.clikt.core.main
import kotlin.test.Test
import okhttp3.TestUtil
import org.junit.jupiter.api.BeforeAll

class OkcurlTest {
@Test
fun simple() {
Main().main(listOf("--help"))
}

companion object {
@JvmStatic
@BeforeAll
fun beforeAll() {
// https://github.com/square/okhttp/issues/8342
TestUtil.assumeNotWindows()
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -176,7 +176,7 @@ class DnsOverHttpsTest {
// 5. unsuccessful response
@Test
fun usesCache() {
val cache = Cache(cacheFs, "cache".toPath(), (100 * 1024).toLong())
val cache = Cache(cacheFs, "cache-usesCache".toPath(), (100 * 1024).toLong())
val cachedClient = bootstrapClient.newBuilder().cache(cache).build()
val cachedDns = buildLocalhost(cachedClient, false)

Expand Down Expand Up @@ -220,7 +220,7 @@ class DnsOverHttpsTest {

@Test
fun usesCacheEvenForPost() {
val cache = Cache(cacheFs, "cache".toPath(), (100 * 1024).toLong())
val cache = Cache(cacheFs, "cache-usesCacheEvenForPost".toPath(), (100 * 1024).toLong())
val cachedClient = bootstrapClient.newBuilder().cache(cache).build()
val cachedDns = buildLocalhost(cachedClient, false, post = true)
repeat(2) {
Expand Down Expand Up @@ -259,6 +259,8 @@ class DnsOverHttpsTest {
.isEqualTo("/lookup?ct")

assertThat(cacheEvents()).containsExactly("CacheMiss")

cache.close()
}

@Test
Expand Down Expand Up @@ -305,6 +307,8 @@ class DnsOverHttpsTest {
.isEqualTo("/lookup?ct&dns=AAABAAABAAAAAAAABmdvb2dsZQNjb20AAAEAAQ")

assertThat(cacheEvents()).containsExactly("CacheMiss")

cache.close()
}

private fun cacheEvents(): List<String> {
Expand Down
1 change: 1 addition & 0 deletions okhttp/build.gradle.kts
Original file line number Diff line number Diff line change
Expand Up @@ -147,6 +147,7 @@ kotlin {
implementation(libs.kotlin.test.junit)
implementation(libs.openjsse)
compileOnly(libs.findbugs.jsr305)
implementation(libs.kotlinx.coroutines.test)

implementation(libs.junit.jupiter.engine)
implementation(libs.junit.vintage.engine)
Expand Down
11 changes: 7 additions & 4 deletions okhttp/src/commonJvmAndroid/kotlin/okhttp3/Cache.kt
Original file line number Diff line number Diff line change
Expand Up @@ -150,17 +150,19 @@ class Cache internal constructor(
maxSize: Long,
fileSystem: FileSystem,
taskRunner: TaskRunner,
useCacheLock: Boolean,
) : Closeable, Flushable {
/** Create a cache of at most [maxSize] bytes in [directory]. */
constructor(
fileSystem: FileSystem,
directory: Path,
maxSize: Long,
) : this(
directory,
maxSize,
fileSystem,
TaskRunner.INSTANCE,
directory = directory,
maxSize = maxSize,
fileSystem = fileSystem,
taskRunner = TaskRunner.INSTANCE,
useCacheLock = true,
)

/** Create a cache of at most [maxSize] bytes in [directory]. */
Expand All @@ -178,6 +180,7 @@ class Cache internal constructor(
valueCount = ENTRY_COUNT,
maxSize = maxSize,
taskRunner = taskRunner,
useCacheLock = useCacheLock,
)

// read and write statistics, all guarded by 'this'.
Expand Down
115 changes: 115 additions & 0 deletions okhttp/src/commonJvmAndroid/kotlin/okhttp3/internal/cache/CacheLock.kt
Original file line number Diff line number Diff line change
@@ -0,0 +1,115 @@
/*
* Copyright (C) 2024 Block, Inc.
*
* 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 okhttp3.internal.cache

import java.io.Closeable
import java.io.File
import java.nio.channels.FileChannel
import java.nio.file.StandardOpenOption
import java.util.Collections
import okio.FileSystem
import okio.Path
import okio.Path.Companion.toOkioPath
import org.codehaus.mojo.animal_sniffer.IgnoreJRERequirement

/**
* An implementation of directory locking to ensure exclusive access in a Cache instance.
* Always applies for the current process, and uses a file system lock if supported.
*/
internal object CacheLock {
private val openCaches = Collections.synchronizedMap(mutableMapOf<Path, Exception>())

/**
* Open a lock, which if successful remains until the returned [Closeable] is closed.
* The lock file will be a file with name "lock" inside directory.
*
* @param fileSystem the file system containing the lock files.
* @param directory the cache directory.
*/
fun openLock(
fileSystem: FileSystem,
directory: Path,
): Closeable {
val memoryLock = inMemoryLock(directory)

// check if possibly a non System file system
if (FileSystem.SYSTEM.exists(directory)) {
try {
val fileSystemLock = fileSystemLock(directory)

return Closeable {
memoryLock.close()
fileSystemLock.close()
}
} catch (le: LockException) {
if (fileSystemSupportsLock(fileSystem)) {
memoryLock.close()
throw le
}
}
}

return memoryLock
}

@IgnoreJRERequirement // Not called on legacy Android
internal fun fileSystemSupportsLock(fileSystem: FileSystem): Boolean {
val tmpLockFile = File.createTempFile("test-", ".lock")

if (!fileSystem.exists(tmpLockFile.toOkioPath())) {
return false
}

val channel = FileChannel.open(tmpLockFile.toPath(), StandardOpenOption.APPEND)

return channel.tryLock().apply { close() } != null
}

/**
* Create an in-memory lock, avoiding two open Cache instances.
*/
@IgnoreJRERequirement // D8 supports put if absent
internal fun inMemoryLock(directory: Path): Closeable {
val existing = openCaches.putIfAbsent(directory, LockException("Existing CacheLock($directory) opened at"))
if (existing != null) {
throw LockException("Cache already open at '$directory' in same process", existing)
}
return okio.Closeable {
openCaches.remove(directory)
}
}

/**
* Create a file system lock, that excludes other processes. However within the process a
* memory lock is also needed, since locks don't work within a single process.
*/
@IgnoreJRERequirement // Not called on legacy Android
internal fun fileSystemLock(directory: Path): Closeable {
// update once https://github.com/square/okio/issues/1464 is available

val lockFile = directory / "lock"
lockFile.toFile().createNewFile()
val channel = FileChannel.open(lockFile.toNioPath(), StandardOpenOption.APPEND)

channel.tryLock() ?: throw LockException("Cache already open at '$directory' in another process")

return okio.Closeable {
channel.close()
}
}
}

class LockException(message: String, cause: Exception? = null) : Exception(message, cause)
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,7 @@ import java.io.EOFException
import java.io.Flushable
import java.io.IOException
import okhttp3.internal.assertThreadHoldsLock
import okhttp3.internal.cache.CacheLock.openLock
import okhttp3.internal.cache.DiskLruCache.Editor
import okhttp3.internal.closeQuietly
import okhttp3.internal.concurrent.Task
Expand Down Expand Up @@ -94,7 +95,10 @@ class DiskLruCache(
maxSize: Long,
/** Used for asynchronous journal rebuilds. */
taskRunner: TaskRunner,
private val useCacheLock: Boolean,
) : Closeable, Flushable {
lateinit var cacheLock: Closeable

internal val fileSystem: FileSystem =
object : ForwardingFileSystem(fileSystem) {
override fun sink(
Expand Down Expand Up @@ -242,33 +246,42 @@ class DiskLruCache(

civilizedFileSystem = fileSystem.isCivilized(journalFileBackup)

// Prefer to pick up where we left off.
if (fileSystem.exists(journalFile)) {
try {
readJournal()
processJournal()
initialized = true
return
} catch (journalIsCorrupt: IOException) {
Platform.get().log(
"DiskLruCache $directory is corrupt: ${journalIsCorrupt.message}, removing",
WARN,
journalIsCorrupt,
)
}
cacheLock = if (useCacheLock) openLock(fileSystem, directory) else Closeable {}

// The cache is corrupted, attempt to delete the contents of the directory. This can throw and
// we'll let that propagate out as it likely means there is a severe filesystem problem.
try {
delete()
} finally {
closed = false
try {
// Prefer to pick up where we left off.
if (fileSystem.exists(journalFile)) {
try {
readJournal()
processJournal()
initialized = true
return
} catch (journalIsCorrupt: IOException) {
Platform.get().log(
"DiskLruCache $directory is corrupt: ${journalIsCorrupt.message}, removing",
WARN,
journalIsCorrupt,
)
}

// The cache is corrupted, attempt to delete the contents of the directory. This can throw and
// we'll let that propagate out as it likely means there is a severe filesystem problem.
try {
delete()
} finally {
closed = false
}
}
}

rebuildJournal()
rebuildJournal()

initialized = true
initialized = true
} finally {
// If anything failed, leave without a cache lock open
if (!initialized) {
cacheLock.close()
}
}
}

@Throws(IOException::class)
Expand Down Expand Up @@ -705,6 +718,8 @@ class DiskLruCache(
return
}

cacheLock.close()

// Copying for concurrent iteration.
for (entry in lruEntries.values.toTypedArray()) {
if (entry.currentEditor != null) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -344,7 +344,10 @@ class Relay private constructor(
val header = Buffer()
fileOperator.read(0, header, FILE_HEADER_SIZE)
val prefix = header.readByteString(PREFIX_CLEAN.size.toLong())
if (prefix != PREFIX_CLEAN) throw IOException("unreadable cache file")
if (prefix != PREFIX_CLEAN) {
randomAccessFile.close()
throw IOException("unreadable cache file")
}
val upstreamSize = header.readLong()
val metadataSize = header.readLong()

Expand Down
Loading
Loading