commonMain.aws.smithy.kotlin.runtime.awsprotocol.eventstream.Message.kt Maven / Gradle / Ivy
/*
* Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
* SPDX-License-Identifier: Apache-2.0
*/
package aws.smithy.kotlin.runtime.awsprotocol.eventstream
import aws.smithy.kotlin.runtime.InternalApi
import aws.smithy.kotlin.runtime.hashing.Crc32
import aws.smithy.kotlin.runtime.io.*
import aws.smithy.kotlin.runtime.text.encoding.encodeToHex
internal const val MESSAGE_CRC_BYTE_LEN = 4
// max message size is 16 MB
internal const val MAX_MESSAGE_SIZE = 16 * 1024 * 1024
// max header size is 128 KB
internal const val MAX_HEADER_SIZE = 128 * 1024
/*
Message Wire Format
See also: https://docs.aws.amazon.com/transcribe/latest/dg/event-stream-med.html
+--------------------------------------------------------------------+ --
| Total Len (32) | |
+--------------------------------------------------------------------+ | Prelude
| Headers Len (32) | |
+--------------------------------------------------------------------+ |
| Prelude CRC (32) | |
+--------------------------------------------------------------------+ --
| Headers (*) ... |
+--------------------------------------------------------------------+
| Payload (*) ... |
+--------------------------------------------------------------------+
| Message CRC (32) |
+--------------------------------------------------------------------+
*/
/**
* An event stream message
*/
@InternalApi
public data class Message(val headers: List, val payload: ByteArray) {
@InternalApi
public companion object {
/**
* Read a message from [source]
*/
public fun decode(source: SdkBufferedSource): Message {
val totalLen = source.peek().use { it.readInt().toUInt() }
check(totalLen <= MAX_MESSAGE_SIZE.toUInt()) { "Invalid Message size: $totalLen" }
// Limiting the amount of data read by SdkBufferedSource is tricky and cause incorrect CRC
// if not careful (e.g. creating a buffered source of a HashingSource will usually lead to incorrect results
// because the entire point SdkBufferedSource (okio.BufferedSource) is to buffer larger chunks internally
// to optimize short reads)
val messageBuffer = SdkBuffer()
val computedCrc = run {
val crcSource = HashingSource(Crc32(), source)
crcSource.readFully(messageBuffer, totalLen.toLong() - MESSAGE_CRC_BYTE_LEN.toLong())
crcSource.digest()
}
val prelude = Prelude.decode(messageBuffer)
val remaining = prelude.totalLen - PRELUDE_BYTE_LEN_WITH_CRC - MESSAGE_CRC_BYTE_LEN
check(messageBuffer.request(remaining.toLong())) { "Invalid buffer, not enough remaining; have: ${messageBuffer.size}; expected $remaining" }
val message = MessageBuilder()
// read headers
var headerBytesConsumed = 0L
while (headerBytesConsumed < prelude.headersLength.toLong()) {
val start = messageBuffer.buffer.size
val header = Header.decode(messageBuffer)
headerBytesConsumed += start - messageBuffer.buffer.size
message.addHeader(header)
}
check(headerBytesConsumed == prelude.headersLength.toLong()) { "Invalid Message: expected ${prelude.headersLength} header bytes; consumed $headerBytesConsumed" }
message.payload = messageBuffer.readByteArray(prelude.payloadLen.toLong())
val expectedCrc = source.readByteArray(4)
check(computedCrc.contentEquals(expectedCrc)) {
"Message checksum mismatch; expected=0x${expectedCrc.encodeToHex()}; calculated=0x${computedCrc.encodeToHex()}"
}
return message.build()
}
}
override fun equals(other: Any?): Boolean {
if (this === other) return true
if (other == null || this::class != other::class) return false
other as Message
if (headers != other.headers) return false
if (!payload.contentEquals(other.payload)) return false
return true
}
override fun hashCode(): Int {
var result = headers.hashCode()
result = 31 * result + payload.contentHashCode()
return result
}
/**
* Encode a message to the [dest] buffer
*/
public fun encode(dest: SdkBufferedSink) {
val headerBuf = SdkBuffer()
headers.forEach { it.encode(headerBuf) }
val headersLen = headerBuf.size
val payloadLen = payload.size
val messageLen = PRELUDE_BYTE_LEN_WITH_CRC + headersLen + payloadLen + MESSAGE_CRC_BYTE_LEN
check(headersLen < MAX_HEADER_SIZE) { "Invalid Headers length: $headersLen" }
check(messageLen < MAX_MESSAGE_SIZE) { "Invalid Message length: $messageLen" }
val prelude = Prelude(messageLen.toInt(), headersLen.toInt())
val sink = HashingSink(Crc32(), dest)
val buffer = sink.buffer()
prelude.encode(buffer)
buffer.write(headerBuf, headerBuf.size)
buffer.write(payload)
buffer.emit()
dest.write(sink.digest())
}
}
private fun emptyByteArray(): ByteArray = ByteArray(0)
/**
* Used to constructing a single event stream [Message]
*/
@InternalApi
public class MessageBuilder {
public val headers: MutableList = mutableListOf()
public var payload: ByteArray? = null
public fun addHeader(header: Header) {
headers.add(header)
}
public fun addHeader(name: String, value: HeaderValue) {
headers.add(Header(name, value))
}
public fun build(): Message = Message(headers, payload ?: emptyByteArray())
}
/**
* Builds a new [Message] by populating a [MessageBuilder] using the given [block]
* @return the constructed messsage
*/
@InternalApi
public fun buildMessage(block: MessageBuilder.() -> Unit): Message = MessageBuilder().apply(block).build()
© 2015 - 2025 Weber Informatics LLC | Privacy Policy