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

kinesis4cats.MD5.scala Maven / Gradle / Ivy

/*
 * Copyright 2023-2023 etspaceman
 *
 * 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 kinesis4cats

/** MessageDigest is incredibly heavy-weight and has no linkages in ScalaJS. We
  * could use the FS2 MD5 hashing, but that spins up MessageDigest with each
  * pass, which leads to really bad performance. So I created this to deal with
  * MD5 hashing in a performant way.
  *
  * See https://rosettacode.org/wiki/MD5/Implementation#Java
  */
@SuppressWarnings(Array("scalafix:DisableSyntax.var"))
private[kinesis4cats] object MD5 {
  private val initA: Int = 0x67452301
  private val initB: Int = 0xefcdab89L.toInt
  private val initC: Int = 0x98badcfeL.toInt
  private val initD: Int = 0x10325476
  private val shiftAmts: Array[Int] =
    Array(7, 12, 17, 22, 5, 9, 14, 20, 4, 11, 16, 23, 6, 10, 15, 21)

  private val tableT: Array[Int] = Array.ofDim(64)
  for (i <- 0 until 64)
    tableT(i) = ((1L << 32) * Math.abs(Math.sin((i + 1).toDouble))).toLong.toInt

  def compute(message: Array[Byte]): Array[Byte] = {
    val messageLenBytes = message.length
    val numBlocks = ((messageLenBytes + 8) >>> 6) + 1
    val totalLen = numBlocks << 6
    val paddingBytes: Array[Byte] = Array.ofDim(totalLen - messageLenBytes)
    paddingBytes(0) = 0x80.toByte

    var messageLenBits = messageLenBytes.toLong << 3
    for (i <- 0 until 8) {
      paddingBytes(paddingBytes.length - 8 + i) = messageLenBits.toByte
      messageLenBits = messageLenBits >>> 8
    }
    var a = initA
    var b = initB
    var c = initC
    var d = initD
    val buffer: Array[Int] = Array.ofDim(16)

    for (i <- 0 until numBlocks) {
      var index = i << 6

      for (j <- 0 until 64) {
        buffer(j >>> 2) = ((if (index < messageLenBytes) message(index)
                            else
                              paddingBytes(
                                index - messageLenBytes
                              )) << 24).toInt | (buffer(j >>> 2) >>> 8)
        index = index + 1
      }
      val originalA = a
      val originalB = b
      val originalC = c
      val originalD = d

      for (j <- 0 until 64) {
        val div16 = j >>> 4
        var f = 0
        var bufferIndex = j
        div16 match {
          case 0 =>
            f = (b & c) | (~b & d)
          case 1 =>
            f = (b & d) | (c & ~d)
            bufferIndex = (bufferIndex * 5 + 1) & 0x0f
          case 2 =>
            f = b ^ c ^ d
            bufferIndex = (bufferIndex * 3 + 5) & 0x0f
          case 3 =>
            f = c ^ (b | ~d)
            bufferIndex = (bufferIndex * 7) & 0x0f
          case _ => ()
        }
        val temp = b + Integer.rotateLeft(
          a + f + buffer(bufferIndex) + tableT(j),
          shiftAmts((div16 << 2) | (j & 3))
        )
        a = d
        d = c
        c = b
        b = temp
      }

      a = a + originalA
      b = b + originalB
      c = c + originalC
      d = d + originalD
    }

    val md5: Array[Byte] = Array.ofDim(16)
    var count = 0

    for (i <- 0 until 4) {
      var n = i match {
        case 0 => a
        case 1 => b
        case 2 => c
        case _ => d
      }

      for (_ <- 0 until 4) {
        md5(count) = n.toByte
        count = count + 1
        n = n >>> 8
      }
    }

    md5
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy