commonMain.io.ktor.util.ByteChannels.kt Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of ktor-utils-watchosarm64 Show documentation
Show all versions of ktor-utils-watchosarm64 Show documentation
Ktor is a framework for quickly creating web applications in Kotlin with minimal effort.
The newest version!
/*
* Copyright 2014-2021 JetBrains s.r.o and contributors. Use of this source code is governed by the Apache 2.0 license.
*/
package io.ktor.util
import io.ktor.utils.io.*
import io.ktor.utils.io.core.*
import io.ktor.utils.io.pool.*
import kotlinx.coroutines.*
private const val CHUNK_BUFFER_SIZE = 4096L
/**
* Split source [ByteReadChannel] into 2 new ones.
* Cancel of one channel in split (input or both outputs) cancels other channels.
*/
public fun ByteReadChannel.split(coroutineScope: CoroutineScope): Pair {
val first = ByteChannel(autoFlush = true)
val second = ByteChannel(autoFlush = true)
coroutineScope.launch {
val buffer = ByteArrayPool.borrow()
try {
while (!isClosedForRead) {
val read = [email protected](buffer)
if (read <= 0) continue
listOf(
async { first.writeFully(buffer, 0, read) },
async { second.writeFully(buffer, 0, read) }
).awaitAll()
}
closedCause?.let { throw it }
} catch (cause: Throwable) {
[email protected](cause)
first.cancel(cause)
second.cancel(cause)
} finally {
ByteArrayPool.recycle(buffer)
first.close()
second.close()
}
}.invokeOnCompletion {
it ?: return@invokeOnCompletion
first.cancel(it)
second.cancel(it)
}
return first to second
}
/**
* Copy a source channel to both output channels chunk by chunk.
*/
@OptIn(DelicateCoroutinesApi::class)
public fun ByteReadChannel.copyToBoth(first: ByteWriteChannel, second: ByteWriteChannel) {
GlobalScope.launch(Dispatchers.Default) {
try {
while (!isClosedForRead && (!first.isClosedForWrite || !second.isClosedForWrite)) {
readRemaining(CHUNK_BUFFER_SIZE).use {
try {
first.writePacket(it.copy())
second.writePacket(it.copy())
} catch (cause: Throwable) {
[email protected](cause)
first.close(cause)
second.close(cause)
}
}
}
closedCause?.let { throw it }
} catch (cause: Throwable) {
first.close(cause)
second.close(cause)
} finally {
first.flushAndClose()
second.flushAndClose()
}
}.invokeOnCompletion {
it ?: return@invokeOnCompletion
first.close(it)
second.close(it)
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy