2019-09-23 06:43:12 +02:00
|
|
|
import Curve25519Kit
|
2019-09-19 08:05:27 +02:00
|
|
|
import PromiseKit
|
|
|
|
|
2019-09-20 06:26:29 +02:00
|
|
|
@objc (LKDeviceLinkingSession)
|
2019-09-20 07:53:24 +02:00
|
|
|
public final class LokiDeviceLinkingSession : NSObject {
|
2019-09-19 08:05:27 +02:00
|
|
|
private let delegate: LokiDeviceLinkingSessionDelegate
|
2019-09-20 06:26:29 +02:00
|
|
|
@objc public var isListeningForLinkingRequests = false
|
2019-09-19 08:05:27 +02:00
|
|
|
|
|
|
|
// MARK: Lifecycle
|
2019-09-23 05:42:58 +02:00
|
|
|
@objc public static var current: LokiDeviceLinkingSession?
|
|
|
|
|
|
|
|
private init(delegate: LokiDeviceLinkingSessionDelegate) {
|
2019-09-19 08:05:27 +02:00
|
|
|
self.delegate = delegate
|
|
|
|
}
|
|
|
|
|
|
|
|
// MARK: Public API
|
2019-09-23 05:42:58 +02:00
|
|
|
public static func startListeningForLinkingRequests(with delegate: LokiDeviceLinkingSessionDelegate) -> LokiDeviceLinkingSession {
|
|
|
|
let session = LokiDeviceLinkingSession(delegate: delegate)
|
|
|
|
session.isListeningForLinkingRequests = true
|
|
|
|
LokiDeviceLinkingSession.current = session
|
|
|
|
return session
|
2019-09-19 08:05:27 +02:00
|
|
|
}
|
|
|
|
|
2019-09-20 06:26:29 +02:00
|
|
|
@objc public func processLinkingRequest(from slaveHexEncodedPublicKey: String, with slaveSignature: Data) {
|
2019-09-20 02:59:20 +02:00
|
|
|
guard isListeningForLinkingRequests else { return }
|
2019-09-23 02:35:31 +02:00
|
|
|
stopListeningForLinkingRequests()
|
2019-09-20 02:59:20 +02:00
|
|
|
let master = LokiDeviceLink.Device(hexEncodedPublicKey: OWSIdentityManager.shared().identityKeyPair()!.hexEncodedPublicKey)
|
|
|
|
let slave = LokiDeviceLink.Device(hexEncodedPublicKey: slaveHexEncodedPublicKey, signature: slaveSignature)
|
|
|
|
let deviceLink = LokiDeviceLink(between: master, and: slave)
|
2019-09-24 01:21:34 +02:00
|
|
|
guard isValidLinkingRequest(deviceLink) else { return }
|
2019-09-20 06:08:35 +02:00
|
|
|
delegate.requestUserAuthorization(for: deviceLink)
|
|
|
|
}
|
|
|
|
|
2019-09-23 05:42:58 +02:00
|
|
|
public func stopListeningForLinkingRequests() {
|
|
|
|
LokiDeviceLinkingSession.current = nil
|
2019-09-23 02:35:31 +02:00
|
|
|
isListeningForLinkingRequests = false
|
|
|
|
}
|
|
|
|
|
2019-09-23 05:42:58 +02:00
|
|
|
public func authorizeDeviceLink(_ deviceLink: LokiDeviceLink) {
|
2019-09-20 06:08:35 +02:00
|
|
|
// TODO: Send a device link authorized message
|
2019-09-20 02:59:20 +02:00
|
|
|
}
|
|
|
|
|
2019-09-20 06:08:35 +02:00
|
|
|
// MARK: Private API
|
2019-09-24 01:21:34 +02:00
|
|
|
private func isValidLinkingRequest(_ deviceLink: LokiDeviceLink) -> Bool {
|
|
|
|
// When requesting a device link, the slave device signs the master device's public key. When authorizing
|
|
|
|
// a device link, the master device signs the slave device's public key.
|
|
|
|
let slaveSignature = deviceLink.slave.signature!
|
|
|
|
let slavePublicKey = Data(hex: deviceLink.slave.hexEncodedPublicKey)
|
|
|
|
let masterPublicKey = Data(hex: deviceLink.master.hexEncodedPublicKey)
|
|
|
|
return (try? Ed25519.verifySignature(slaveSignature, publicKey: slavePublicKey, data: masterPublicKey)) ?? false
|
2019-09-20 06:08:35 +02:00
|
|
|
}
|
2019-09-19 08:05:27 +02:00
|
|
|
}
|