mirror of https://github.com/oxen-io/session-ios
You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
39 lines
1.9 KiB
Swift
39 lines
1.9 KiB
Swift
5 years ago
|
import CryptoSwift
|
||
|
|
||
|
internal typealias EncryptionResult = (ciphertext: Data, symmetricKey: Data, ephemeralPublicKey: Data)
|
||
|
|
||
|
enum EncryptionUtilities {
|
||
|
internal static let gcmTagSize: UInt = 16
|
||
|
internal static let ivSize: UInt = 12
|
||
|
|
||
|
/// - Note: Sync. Don't call from the main thread.
|
||
|
internal static func encrypt(_ plaintext: Data, usingAESGCMWithSymmetricKey symmetricKey: Data) throws -> Data {
|
||
|
if Thread.isMainThread {
|
||
|
#if DEBUG
|
||
|
preconditionFailure("It's illegal to call encrypt(_:usingAESGCMWithSymmetricKey:) from the main thread.")
|
||
|
#endif
|
||
|
}
|
||
|
let iv = Data.getSecureRandomData(ofSize: ivSize)!
|
||
|
let gcm = GCM(iv: iv.bytes, tagLength: Int(gcmTagSize), mode: .combined)
|
||
|
let aes = try AES(key: symmetricKey.bytes, blockMode: gcm, padding: .noPadding)
|
||
|
let ciphertext = try aes.encrypt(plaintext.bytes)
|
||
|
return iv + Data(bytes: ciphertext)
|
||
|
}
|
||
|
|
||
|
/// - Note: Sync. Don't call from the main thread.
|
||
|
internal static func encrypt(_ plaintext: Data, using hexEncodedX25519PublicKey: String) throws -> EncryptionResult {
|
||
|
if Thread.isMainThread {
|
||
|
#if DEBUG
|
||
|
preconditionFailure("It's illegal to call encrypt(_:forSnode:) from the main thread.")
|
||
|
#endif
|
||
|
}
|
||
|
let x25519PublicKey = Data(hex: hexEncodedX25519PublicKey)
|
||
|
let ephemeralKeyPair = Curve25519.generateKeyPair()!
|
||
|
let ephemeralSharedSecret = Curve25519.generateSharedSecret(fromPublicKey: x25519PublicKey, andKeyPair: ephemeralKeyPair)!
|
||
|
let salt = "LOKI"
|
||
|
let symmetricKey = try HMAC(key: salt.bytes, variant: .sha256).authenticate(ephemeralSharedSecret.bytes)
|
||
|
let ciphertext = try encrypt(plaintext, usingAESGCMWithSymmetricKey: Data(bytes: symmetricKey))
|
||
|
return (ciphertext, Data(bytes: symmetricKey), ephemeralKeyPair.publicKey())
|
||
|
}
|
||
|
}
|