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

smithy4s.Blob.scala Maven / Gradle / Ivy

/*
 *  Copyright 2021-2024 Disney Streaming
 *
 *  Licensed under the Tomorrow Open Source Technology License, Version 1.0 (the "License");
 *  you may not use this file except in compliance with the License.
 *  You may obtain a copy of the License at
 *
 *     https://disneystreaming.github.io/TOST-1.0.txt
 *
 *  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 smithy4s

import java.nio.ByteBuffer
import java.util.Base64
import scala.collection.immutable.Queue
import java.nio.charset.StandardCharsets
import java.io.OutputStream

// scalafmt: {maxColumn = 120}
/**
  * A Blob represents an arbitrary piece of binary data that fits in memory.
  *
  * Its underlying data structure enables several types of layouts, as well as efficient concatenation.
  */
sealed trait Blob {

  def apply(i: Int): Byte
  def size: Int
  def isEmpty: Boolean

  def foreach(f: Byte => Unit) = {
    var i = 0
    while (i < size) { f(apply(i)); i += 1 }
  }

  def foreachWithIndex(f: (Byte, Int) => Unit) = {
    var i = 0
    while (i < size) { f(apply(i), i); i += 1 }
  }

  def toArraySliceBlob: Blob.ArraySliceBlob =
    new Blob.ArraySliceBlob(toArrayUnsafe, 0, size)

  def toArray: Array[Byte] = {
    val result = Array.ofDim[Byte](size)
    foreachWithIndex((b, i) => result(i) = b)
    result
  }

  def toArrayUnsafe: Array[Byte] = toArray

  def sameBytesAs(other: Blob): Boolean = {
    size == other.size && {
      var i = 0
      var result = true
      while (i < size && result) {
        result = this(i) == other(i)
        i += 1
      }
      result
    }
  }

  def asByteBuffer(offset: Int, size: Int): ByteBuffer = {
    val arr = new Array[Byte](size)
    copyToArray(arr, 0, offset, size)
    ByteBuffer.wrap(arr)
  }
  def asByteBuffer: ByteBuffer = asByteBuffer(0, size)
  def asByteBufferUnsafe(offset: Int, size: Int): ByteBuffer = asByteBuffer(offset, size)
  def asByteBufferUnsafe: ByteBuffer = asByteBuffer(0, size)

  def copyToArray(xs: Array[Byte], start: Int, offset: Int, size: Int): Unit = {
    var i = 0
    while (i < size) {
      xs(start + i) = apply(offset + i)
      i += 1
    }
  }

  def copyToBuffer(buffer: ByteBuffer, offset: Int, size: Int): Int = {
    var i = 0
    while (i < size && buffer.remaining > 0) {
      buffer.put(apply(offset + i))
      i += 1
    }
    i
  }

  def copyToStream(s: OutputStream, offset: Int, size: Int): Unit = {
    var i = 0
    while (i < size) {
      s.write(apply(offset + i).toInt)
      i += 1
    }
  }

  final def toBase64String: String = Base64.getEncoder().encodeToString(toArray)
  final def toUTF8String: String = new String(toArray, StandardCharsets.UTF_8)

  def concat(other: Blob) =
    if (this.isEmpty) other
    else
      this match {
        case qb: Blob.QueueBlob => new Blob.QueueBlob(qb.blobs :+ other, qb.size + other.size)
        case b                  => new Blob.QueueBlob(Queue(b, other), this.size + other.size)
      }

  final def ++(other: Blob) = concat(other)

  override def equals(other: Any): Boolean =
    other match {
      case otherBlob: Blob => sameBytesAs(otherBlob)
      case _               => false
    }

  override def hashCode(): Int = {
    import util.hashing.MurmurHash3
    var h = MurmurHash3.stringHash("Blob")
    foreach(o => h = MurmurHash3.mix(h, o.##))
    MurmurHash3.finalizeHash(h, size)
  }

}

object Blob {

  val empty: Blob = new Blob.ArraySliceBlob(Array.emptyByteArray, 0, 0)

  def apply(bytes: Array[Byte]): Blob = new ArraySliceBlob(bytes, 0, bytes.length)
  def apply(string: String): Blob = apply(string.getBytes(StandardCharsets.UTF_8))
  def apply(buffer: ByteBuffer): Blob = view(buffer.duplicate())

  def slice(bytes: Array[Byte], offset: Int, size: Int): Blob = new ArraySliceBlob(bytes, offset, size)
  def view(buffer: ByteBuffer): Blob = new ByteBufferBlob(buffer)
  def queue(blobs: Queue[Blob], size: Int) = new QueueBlob(blobs, size)

  final class ByteBufferBlob private[smithy4s] (val buf: ByteBuffer) extends Blob {
    def apply(i: Int) = buf.get(i.toInt)

    override def toArraySliceBlob: ArraySliceBlob = if (buf.hasArray()) {
      new ArraySliceBlob(buf.array, buf.arrayOffset, size)
    } else super.toArraySliceBlob

    override def copyToArray(xs: Array[Byte], start: Int, offset: Int, size: Int): Unit = {
      val n = buf.duplicate()
      n.position(offset.toInt)
      n.get(xs, start, size)
      ()
    }

    override def toArray: Array[Byte] = {
      val arr = Array.ofDim[Byte](buf.remaining())
      copyToArray(arr, 0, 0, size)
      arr
    }

    override def asByteBuffer(offset: Int, size: Int): ByteBuffer =
      asByteBufferUnsafe(offset, size).asReadOnlyBuffer()

    override def asByteBufferUnsafe(offset: Int, size: Int): ByteBuffer = {
      val b = buf.duplicate()
      if (offset == 0 && b.position() == 0 && size == b.remaining()) b
      else {
        b.position(offset.toInt)
        b.limit(offset.toInt + size)
        b.slice()
      }
    }

    override def asByteBufferUnsafe: ByteBuffer = buf.duplicate()

    override def copyToBuffer(buffer: ByteBuffer, offset: Int, size: Int): Int = {
      val toCopy = buffer.remaining.min(size)
      buffer.put(asByteBuffer(offset, toCopy))
      toCopy
    }

    override def toString = s"ByteBufferBlob(...)"
    override def isEmpty: Boolean = !buf.hasRemaining()
    override def size: Int = buf.remaining()

  }

  final class ArraySliceBlob private[smithy4s] (val arr: Array[Byte], val offset: Int, val length: Int) extends Blob {

    override def toArraySliceBlob: ArraySliceBlob = this

    require(
      offset >= 0 && offset <= arr.size && length >= 0 && length <= arr.size && offset + length <= arr.size
    )

    def apply(i: Int): Byte = {
      if (i >= length) {
        throw new IndexOutOfBoundsException()
      } else arr(offset + i)
    }

    def size: Int = length
    def isEmpty: Boolean = (length == 0)

    override def toArray: Array[Byte] = {
      val result = Array.ofDim[Byte](length)
      arr.copyToArray(result, offset, length)
      result
    }

    override def toArrayUnsafe: Array[Byte] = if (arr.length == length && offset == 0) arr else toArray

    override def toString(): String = s"ArraySliceBlob(..., $offset, $length)"

  }

  final class QueueBlob private[smithy4s] (val blobs: Queue[Blob], val size: Int) extends Blob {
    def apply(i: Int): Byte = {
      if (i >= size) throw new IndexOutOfBoundsException()
      else {
        var localIndex = i
        var (currentHead, currentTail) = blobs.dequeue
        while (localIndex >= currentHead.size) {
          localIndex = localIndex - currentHead.size
          val dq = currentTail.dequeue
          currentHead = dq._1
          currentTail = dq._2
        }
        currentHead(localIndex)
      }
    }
    override def foreach(f: Byte => Unit): Unit =
      blobs.foreach(
        _.foreach(f)
      )
    override def foreachWithIndex(f: (Byte, Int) => Unit): Unit = {
      var i = 0
      blobs.foreach { blob =>
        blob.foreach { byte => f(byte, i); i = i + 1 }
      }
    }
    def isEmpty: Boolean = size == 0

    override def toString(): String = s"QueueBlob(..., $size)"
  }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy