All Downloads are FREE. Search and download functionalities are using the official Maven repository.

commonMain.aws.sdk.kotlin.services.glacier.internal.TreeHasher.kt Maven / Gradle / Ivy

/*
 * Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
 * SPDX-License-Identifier: Apache-2.0
 */

package aws.sdk.kotlin.services.glacier.internal

import aws.smithy.kotlin.runtime.hashing.HashSupplier
import aws.smithy.kotlin.runtime.hashing.hash
import aws.smithy.kotlin.runtime.http.HttpBody
import aws.smithy.kotlin.runtime.io.SdkBuffer
import aws.smithy.kotlin.runtime.io.buffer
import kotlinx.coroutines.flow.*
import kotlin.math.min

/**
 * The result of a [TreeHasher] calculation.
 * @param fullHash A full hash of the entire byte array (taken at once)
 * @param treeHash A composite hash of the byte array, taken in chunks.
 */
internal class Hashes(public val fullHash: ByteArray, public val treeHash: ByteArray)

/**
 * A hash calculator that returns [Hashes] derived using a tree. See
 * [Computing Checksums](https://docs.aws.amazon.com/amazonglacier/latest/dev/checksum-calculations.html) in the Glacier
 * service guide for more details.
 */
internal interface TreeHasher {
    /**
     * Perform the hash calculation.
     * @param body The [HttpBody] over which to calculate hashes.
     * @return A [Hashes] containing the results of the calculation.
     */
    suspend fun calculateHashes(body: HttpBody): Hashes
}

/**
 * The default implementation of a [TreeHasher].
 */
internal class TreeHasherImpl(private val chunkSizeBytes: Int, private val hashSupplier: HashSupplier) : TreeHasher {
    override suspend fun calculateHashes(body: HttpBody): Hashes {
        val full = hashSupplier()
        val hashTree = ArrayDeque()

        body.chunks().collect { chunk ->
            full.update(chunk)
            hashTree.addLast(chunk.hash(hashSupplier))
        }

        if (hashTree.isEmpty()) {
            // Edge case for empty bodies
            hashTree.add(byteArrayOf().hash(hashSupplier))
        }

        while (hashTree.size > 1) {
            val nextRow = mutableListOf()

            while (hashTree.isNotEmpty()) {
                if (hashTree.size == 1) {
                    nextRow.add(hashTree.removeFirst())
                } else {
                    val hash = hashSupplier()
                    hashTree.removeFirst().let(hash::update)
                    hashTree.removeFirst().let(hash::update)
                    nextRow.add(hash.digest())
                }
            }

            hashTree.addAll(nextRow)
        }

        return Hashes(full.digest(), hashTree.first())
    }

    private suspend fun HttpBody.chunks(): Flow = when (this) {
        is HttpBody.Empty -> flowOf()

        is HttpBody.Bytes -> {
            val size = bytes().size
            val chunkStarts = 0 until size step chunkSizeBytes
            val chunkRanges = chunkStarts.map { it until min(it + chunkSizeBytes, size) }
            chunkRanges.asFlow().map(bytes()::sliceArray)
        }

        is HttpBody.ChannelContent -> flow {
            val channel = readFrom()
            val sink = SdkBuffer()
            while (!channel.isClosedForRead) {
                var remaining = chunkSizeBytes.toLong()
                while (remaining > 0L) {
                    val rc = channel.read(sink, remaining)
                    if (rc == -1L) break // channel closed before a full chunk could be read
                    remaining -= rc
                }
                emit(sink.readByteArray())
            }
        }
        is HttpBody.SourceContent -> flow {
            val source = readFrom().buffer()
            while (!source.exhausted()) {
                source.request(chunkSizeBytes.toLong())
                val limit = minOf(chunkSizeBytes.toLong(), source.buffer.size)
                emit(source.readByteArray(limit))
            }
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy