MD5.swift 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130
  1. //
  2. // MD5.swift
  3. // Swifter
  4. //
  5. // Copyright © 2016 Damian Kołakowski. All rights reserved.
  6. //
  7. #if os(Linux)
  8. import Glibc
  9. #else
  10. import Foundation
  11. #endif
  12. public struct MD5 {
  13. //
  14. // Alghorithm from: https://en.wikipedia.org/wiki/MD5
  15. //
  16. public static func hash(_ input: [UInt8]) -> [UInt8] {
  17. var a0 = UInt32(littleEndian: 0x67452301)
  18. var b0 = UInt32(littleEndian: 0xefcdab89)
  19. var c0 = UInt32(littleEndian: 0x98badcfe)
  20. var d0 = UInt32(littleEndian: 0x10325476)
  21. var message = input + [0x80] + [UInt8](repeating: 0, count: 64 - ((input.count+9) % 64))
  22. var originalLengthInBits = UInt64(input.count * 8).littleEndian
  23. withUnsafePointer(to: &originalLengthInBits) {
  24. message.append(contentsOf: Array(UnsafeBufferPointer<UInt8>(start: UnsafePointer(OpaquePointer($0)), count: 8)))
  25. }
  26. for chunkStart in 0..<message.count/64 {
  27. var words = [UInt32]()
  28. let chunk = message[chunkStart*64..<chunkStart*64+64]
  29. for i in 0...15 {
  30. words.append(chunk.withUnsafeBufferPointer {
  31. UnsafePointer<UInt32>(OpaquePointer($0.baseAddress! + (i*4))).pointee
  32. })
  33. }
  34. var A = a0, B = b0, C = c0, D = d0
  35. for i in 0...63 {
  36. var F = UInt32(0), g = UInt32(0)
  37. switch i {
  38. case 0...15:
  39. F = (B & C) | ((~B) & D)
  40. g = UInt32(i)
  41. case 16...31:
  42. F = (D & B) | ((~D) & C)
  43. g = UInt32(5*i + 1) % UInt32(16)
  44. case 32...47:
  45. F = B ^ C ^ D
  46. g = UInt32((3*i + 5)) % UInt32(16)
  47. case 48...63:
  48. F = C ^ (B | (~D))
  49. g = UInt32(7*i) % UInt32(16)
  50. default: break
  51. }
  52. let dTemp = D
  53. D = C
  54. C = B
  55. B = B &+ leftrotate((A &+ F &+ K[i] &+ words[Int(g)]), s[i])
  56. A = dTemp
  57. }
  58. a0 = a0 &+ A
  59. b0 = b0 &+ B
  60. c0 = c0 &+ C
  61. d0 = d0 &+ D
  62. }
  63. var digest = [UInt8]()
  64. [a0, b0, c0, d0].forEach { value in
  65. var littleEndianVersion = value.littleEndian
  66. withUnsafePointer(to: &littleEndianVersion) {
  67. digest.append(contentsOf: Array(UnsafeBufferPointer<UInt8>(start: UnsafePointer(OpaquePointer($0)), count: 4)))
  68. }
  69. }
  70. return digest
  71. }
  72. private static func leftrotate(_ v: UInt32, _ n: UInt32) -> UInt32 {
  73. return ((v << n) & 0xFFFFFFFF) | (v >> (32 - n))
  74. }
  75. private static var s: [UInt32] = [
  76. 7, 12, 17, 22, 7, 12, 17, 22, 7, 12, 17, 22, 7, 12, 17, 22,
  77. 5, 9, 14, 20, 5, 9, 14, 20, 5, 9, 14, 20, 5, 9, 14, 20,
  78. 4, 11, 16, 23, 4, 11, 16, 23, 4, 11, 16, 23, 4, 11, 16, 23,
  79. 6, 10, 15, 21, 6, 10, 15, 21, 6, 10, 15, 21, 6, 10, 15, 21
  80. ]
  81. private static var K: [UInt32] = [
  82. 0xd76aa478, 0xe8c7b756, 0x242070db, 0xc1bdceee,
  83. 0xf57c0faf, 0x4787c62a, 0xa8304613, 0xfd469501,
  84. 0x698098d8, 0x8b44f7af, 0xffff5bb1, 0x895cd7be,
  85. 0x6b901122, 0xfd987193, 0xa679438e, 0x49b40821,
  86. 0xf61e2562, 0xc040b340, 0x265e5a51, 0xe9b6c7aa,
  87. 0xd62f105d, 0x02441453, 0xd8a1e681, 0xe7d3fbc8,
  88. 0x21e1cde6, 0xc33707d6, 0xf4d50d87, 0x455a14ed,
  89. 0xa9e3e905, 0xfcefa3f8, 0x676f02d9, 0x8d2a4c8a,
  90. 0xfffa3942, 0x8771f681, 0x6d9d6122, 0xfde5380c,
  91. 0xa4beea44, 0x4bdecfa9, 0xf6bb4b60, 0xbebfbc70,
  92. 0x289b7ec6, 0xeaa127fa, 0xd4ef3085, 0x04881d05,
  93. 0xd9d4d039, 0xe6db99e5, 0x1fa27cf8, 0xc4ac5665,
  94. 0xf4292244, 0x432aff97, 0xab9423a7, 0xfc93a039,
  95. 0x655b59c3, 0x8f0ccc92, 0xffeff47d, 0x85845dd1,
  96. 0x6fa87e4f, 0xfe2ce6e0, 0xa3014314, 0x4e0811a1,
  97. 0xf7537e82, 0xbd3af235, 0x2ad7d2bb, 0xeb86d391
  98. ]
  99. }
  100. extension String {
  101. public func md5() -> String {
  102. return self.md5().hex()
  103. }
  104. public func md5() -> [UInt8] {
  105. return MD5.hash([UInt8](self.utf8))
  106. }
  107. }