123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282 |
- // swiftlint:disable comma function_parameter_count variable_name syntactic_sugar function_body_length vertical_whitespace
- // https://github.com/onmyway133/SwiftHash/blob/master/Sources/MD5.swift
- /*
- * A JavaScript implementation of the RSA Data Security, Inc. MD5 Message
- * Digest Algorithm, as defined in RFC 1321.
- * Version 2.2 Copyright (C) Paul Johnston 1999 - 2009
- * Other contributors: Greg Holt, Andrew Kepert, Ydnar, Lostinet
- * Distributed under the BSD License
- * See http://pajhome.org.uk/crypt/md5 for more info.
- */
- /**
- * SwiftHash
- * Copyright (c) Khoa Pham 2017
- * Licensed under the MIT license. See LICENSE file.
- */
- import Foundation
- // MARK: - Public
- public func MD5(_ input: String) -> String {
- return hex_md5(input)
- }
- // MARK: - Functions
- func hex_md5(_ input: String) -> String {
- return rstr2hex(rstr_md5(str2rstr_utf8(input)))
- }
- func str2rstr_utf8(_ input: String) -> [CUnsignedChar] {
- return Array(input.utf8)
- }
- func rstr2tr(_ input: [CUnsignedChar]) -> String {
- var output: String = ""
- input.forEach {
- output.append(String(UnicodeScalar($0)))
- }
- return output
- }
- /*
- * Convert a raw string to a hex string
- */
- func rstr2hex(_ input: [CUnsignedChar]) -> String {
- let hexTab: [Character] = ["0", "1", "2", "3", "4", "5", "6", "7", "8", "9", "A", "B", "C", "D", "E", "F"]
- var output: [Character] = []
- for i in 0..<input.count {
- let x = input[i]
- let value1 = hexTab[Int((x >> 4) & 0x0F)]
- let value2 = hexTab[Int(Int32(x) & 0x0F)]
- output.append(value1)
- output.append(value2)
- }
- return String(output)
- }
- /*
- * Convert a raw string to an array of little-endian words
- * Characters >255 have their high-byte silently ignored.
- */
- func rstr2binl(_ input: [CUnsignedChar]) -> [Int32] {
- var output: [Int: Int32] = [:]
- for i in stride(from: 0, to: input.count * 8, by: 8) {
- let value: Int32 = (Int32(input[i/8]) & 0xFF) << (Int32(i) % 32)
- output[i >> 5] = unwrap(output[i >> 5]) | value
- }
- return dictionary2array(output)
- }
- /*
- * Convert an array of little-endian words to a string
- */
- func binl2rstr(_ input: [Int32]) -> [CUnsignedChar] {
- var output: [CUnsignedChar] = []
- for i in stride(from: 0, to: input.count * 32, by: 8) {
- // [i>>5] >>>
- let value: Int32 = zeroFillRightShift(input[i>>5], Int32(i % 32)) & 0xFF
- output.append(CUnsignedChar(value))
- }
- return output
- }
- /*
- * Calculate the MD5 of a raw string
- */
- func rstr_md5(_ input: [CUnsignedChar]) -> [CUnsignedChar] {
- return binl2rstr(binl_md5(rstr2binl(input), input.count * 8))
- }
- /*
- * Add integers, wrapping at 2^32. This uses 16-bit operations internally
- * to work around bugs in some JS interpreters.
- */
- func safe_add(_ x: Int32, _ y: Int32) -> Int32 {
- let lsw = (x & 0xFFFF) + (y & 0xFFFF)
- let msw = (x >> 16) + (y >> 16) + (lsw >> 16)
- return (msw << 16) | (lsw & 0xFFFF)
- }
- /*
- * Bitwise rotate a 32-bit number to the left.
- */
- func bit_rol(_ num: Int32, _ cnt: Int32) -> Int32 {
- // num >>>
- return (num << cnt) | zeroFillRightShift(num, (32 - cnt))
- }
- /*
- * These funcs implement the four basic operations the algorithm uses.
- */
- func md5_cmn(_ q: Int32, _ a: Int32, _ b: Int32, _ x: Int32, _ s: Int32, _ t: Int32) -> Int32 {
- return safe_add(bit_rol(safe_add(safe_add(a, q), safe_add(x, t)), s), b)
- }
- func md5_ff(_ a: Int32, _ b: Int32, _ c: Int32, _ d: Int32, _ x: Int32, _ s: Int32, _ t: Int32) -> Int32 {
- return md5_cmn((b & c) | ((~b) & d), a, b, x, s, t)
- }
- func md5_gg(_ a: Int32, _ b: Int32, _ c: Int32, _ d: Int32, _ x: Int32, _ s: Int32, _ t: Int32) -> Int32 {
- return md5_cmn((b & d) | (c & (~d)), a, b, x, s, t)
- }
- func md5_hh(_ a: Int32, _ b: Int32, _ c: Int32, _ d: Int32, _ x: Int32, _ s: Int32, _ t: Int32) -> Int32 {
- return md5_cmn(b ^ c ^ d, a, b, x, s, t)
- }
- func md5_ii(_ a: Int32, _ b: Int32, _ c: Int32, _ d: Int32, _ x: Int32, _ s: Int32, _ t: Int32) -> Int32 {
- return md5_cmn(c ^ (b | (~d)), a, b, x, s, t)
- }
- /*
- * Calculate the MD5 of an array of little-endian words, and a bit length.
- */
- func binl_md5(_ input: [Int32], _ len: Int) -> [Int32] {
- /* append padding */
- var x: [Int: Int32] = [:]
- for (index, value) in input.enumerated() {
- x[index] = value
- }
- let value: Int32 = 0x80 << Int32((len) % 32)
- x[len >> 5] = unwrap(x[len >> 5]) | value
- // >>> 9
- let index = (((len + 64) >> 9) << 4) + 14
- x[index] = unwrap(x[index]) | Int32(len)
- var a: Int32 = 1732584193
- var b: Int32 = -271733879
- var c: Int32 = -1732584194
- var d: Int32 = 271733878
- for i in stride(from: 0, to: length(x), by: 16) {
- let olda: Int32 = a
- let oldb: Int32 = b
- let oldc: Int32 = c
- let oldd: Int32 = d
- a = md5_ff(a, b, c, d, unwrap(x[i + 0]), 7 , -680876936)
- d = md5_ff(d, a, b, c, unwrap(x[i + 1]), 12, -389564586)
- c = md5_ff(c, d, a, b, unwrap(x[i + 2]), 17, 606105819)
- b = md5_ff(b, c, d, a, unwrap(x[i + 3]), 22, -1044525330)
- a = md5_ff(a, b, c, d, unwrap(x[i + 4]), 7 , -176418897)
- d = md5_ff(d, a, b, c, unwrap(x[i + 5]), 12, 1200080426)
- c = md5_ff(c, d, a, b, unwrap(x[i + 6]), 17, -1473231341)
- b = md5_ff(b, c, d, a, unwrap(x[i + 7]), 22, -45705983)
- a = md5_ff(a, b, c, d, unwrap(x[i + 8]), 7 , 1770035416)
- d = md5_ff(d, a, b, c, unwrap(x[i + 9]), 12, -1958414417)
- c = md5_ff(c, d, a, b, unwrap(x[i + 10]), 17, -42063)
- b = md5_ff(b, c, d, a, unwrap(x[i + 11]), 22, -1990404162)
- a = md5_ff(a, b, c, d, unwrap(x[i + 12]), 7 , 1804603682)
- d = md5_ff(d, a, b, c, unwrap(x[i + 13]), 12, -40341101)
- c = md5_ff(c, d, a, b, unwrap(x[i + 14]), 17, -1502002290)
- b = md5_ff(b, c, d, a, unwrap(x[i + 15]), 22, 1236535329)
- a = md5_gg(a, b, c, d, unwrap(x[i + 1]), 5 , -165796510)
- d = md5_gg(d, a, b, c, unwrap(x[i + 6]), 9 , -1069501632)
- c = md5_gg(c, d, a, b, unwrap(x[i + 11]), 14, 643717713)
- b = md5_gg(b, c, d, a, unwrap(x[i + 0]), 20, -373897302)
- a = md5_gg(a, b, c, d, unwrap(x[i + 5]), 5 , -701558691)
- d = md5_gg(d, a, b, c, unwrap(x[i + 10]), 9 , 38016083)
- c = md5_gg(c, d, a, b, unwrap(x[i + 15]), 14, -660478335)
- b = md5_gg(b, c, d, a, unwrap(x[i + 4]), 20, -405537848)
- a = md5_gg(a, b, c, d, unwrap(x[i + 9]), 5 , 568446438)
- d = md5_gg(d, a, b, c, unwrap(x[i + 14]), 9 , -1019803690)
- c = md5_gg(c, d, a, b, unwrap(x[i + 3]), 14, -187363961)
- b = md5_gg(b, c, d, a, unwrap(x[i + 8]), 20, 1163531501)
- a = md5_gg(a, b, c, d, unwrap(x[i + 13]), 5 , -1444681467)
- d = md5_gg(d, a, b, c, unwrap(x[i + 2]), 9 , -51403784)
- c = md5_gg(c, d, a, b, unwrap(x[i + 7]), 14, 1735328473)
- b = md5_gg(b, c, d, a, unwrap(x[i + 12]), 20, -1926607734)
- a = md5_hh(a, b, c, d, unwrap(x[i + 5]), 4 , -378558)
- d = md5_hh(d, a, b, c, unwrap(x[i + 8]), 11, -2022574463)
- c = md5_hh(c, d, a, b, unwrap(x[i + 11]), 16, 1839030562)
- b = md5_hh(b, c, d, a, unwrap(x[i + 14]), 23, -35309556)
- a = md5_hh(a, b, c, d, unwrap(x[i + 1]), 4 , -1530992060)
- d = md5_hh(d, a, b, c, unwrap(x[i + 4]), 11, 1272893353)
- c = md5_hh(c, d, a, b, unwrap(x[i + 7]), 16, -155497632)
- b = md5_hh(b, c, d, a, unwrap(x[i + 10]), 23, -1094730640)
- a = md5_hh(a, b, c, d, unwrap(x[i + 13]), 4 , 681279174)
- d = md5_hh(d, a, b, c, unwrap(x[i + 0]), 11, -358537222)
- c = md5_hh(c, d, a, b, unwrap(x[i + 3]), 16, -722521979)
- b = md5_hh(b, c, d, a, unwrap(x[i + 6]), 23, 76029189)
- a = md5_hh(a, b, c, d, unwrap(x[i + 9]), 4 , -640364487)
- d = md5_hh(d, a, b, c, unwrap(x[i + 12]), 11, -421815835)
- c = md5_hh(c, d, a, b, unwrap(x[i + 15]), 16, 530742520)
- b = md5_hh(b, c, d, a, unwrap(x[i + 2]), 23, -995338651)
- a = md5_ii(a, b, c, d, unwrap(x[i + 0]), 6 , -198630844)
- d = md5_ii(d, a, b, c, unwrap(x[i + 7]), 10, 1126891415)
- c = md5_ii(c, d, a, b, unwrap(x[i + 14]), 15, -1416354905)
- b = md5_ii(b, c, d, a, unwrap(x[i + 5]), 21, -57434055)
- a = md5_ii(a, b, c, d, unwrap(x[i + 12]), 6 , 1700485571)
- d = md5_ii(d, a, b, c, unwrap(x[i + 3]), 10, -1894986606)
- c = md5_ii(c, d, a, b, unwrap(x[i + 10]), 15, -1051523)
- b = md5_ii(b, c, d, a, unwrap(x[i + 1]), 21, -2054922799)
- a = md5_ii(a, b, c, d, unwrap(x[i + 8]), 6 , 1873313359)
- d = md5_ii(d, a, b, c, unwrap(x[i + 15]), 10, -30611744)
- c = md5_ii(c, d, a, b, unwrap(x[i + 6]), 15, -1560198380)
- b = md5_ii(b, c, d, a, unwrap(x[i + 13]), 21, 1309151649)
- a = md5_ii(a, b, c, d, unwrap(x[i + 4]), 6 , -145523070)
- d = md5_ii(d, a, b, c, unwrap(x[i + 11]), 10, -1120210379)
- c = md5_ii(c, d, a, b, unwrap(x[i + 2]), 15, 718787259)
- b = md5_ii(b, c, d, a, unwrap(x[i + 9]), 21, -343485551)
- a = safe_add(a, olda)
- b = safe_add(b, oldb)
- c = safe_add(c, oldc)
- d = safe_add(d, oldd)
- }
- return [a, b, c, d]
- }
- // MARK: - Helper
- func length(_ dictionary: [Int: Int32]) -> Int {
- return (dictionary.keys.max() ?? 0) + 1
- }
- func dictionary2array(_ dictionary: [Int: Int32]) -> [Int32] {
- var array = Array<Int32>(repeating: 0, count: dictionary.keys.count)
- for i in Array(dictionary.keys).sorted() {
- array[i] = unwrap(dictionary[i])
- }
- return array
- }
- func unwrap(_ value: Int32?, _ fallback: Int32 = 0) -> Int32 {
- if let value = value {
- return value
- }
- return fallback
- }
- func zeroFillRightShift(_ num: Int32, _ count: Int32) -> Int32 {
- let value = UInt32(bitPattern: num) >> UInt32(bitPattern: count)
- return Int32(bitPattern: value)
- }
|