Send correct data to storage server.

Data -> Envelope -> WebSocket
pull/12/head
Mikunj 6 years ago
parent 09157673fe
commit 787e2c1cb7

@ -2570,3 +2570,5 @@
"No search results" = "No search results";
"Calculating proof of work" = "Calculating proof of work";
"Failed to calculate proof of work." = "Failed to calculate proof of work.";
"Failed to wrap data in an Envelope" = "Failed to wrap data in an Envelope.";
"Failed to wrap data in an WebSocket" = "Failed to wrap data in an WebSocket.";

@ -21,10 +21,14 @@ import PromiseKit
public enum Error : LocalizedError {
case proofOfWorkCalculationFailed
case failedToWrapInEnvelope
case failedToWrapInWebSocket
public var errorDescription: String? {
switch self {
case .proofOfWorkCalculationFailed: return NSLocalizedString("Failed to calculate proof of work.", comment: "")
case .failedToWrapInEnvelope: return NSLocalizedString("Failed to wrap data in an Envelope", comment: "")
case .failedToWrapInWebSocket: return NSLocalizedString("Failed to wrap data in an WebSocket", comment: "")
}
}
}
@ -73,8 +77,8 @@ import PromiseKit
return anyPromise
}
@objc public static func objc_sendSignalMessage(_ signalMessage: SignalMessage, to destination: String, requiringPoW isPoWRequired: Bool) -> AnyPromise {
let promise = LokiMessage.fromSignalMessage(signalMessage, requiringPoW: isPoWRequired)
@objc public static func objc_sendSignalMessage(_ signalMessage: SignalMessage, to destination: String, timestamp: UInt64, requiringPoW isPoWRequired: Bool) -> AnyPromise {
let promise = LokiMessage.from(signalMessage: signalMessage, timestamp: timestamp, requiringPoW: isPoWRequired)
.then(sendMessage)
.recoverNetworkError(on: DispatchQueue.global())
let anyPromise = AnyPromise(promise)

@ -22,17 +22,40 @@ public struct LokiMessage {
self.nonce = nonce
}
public static func fromSignalMessage(_ signalMessage: SignalMessage, requiringPoW isPoWRequired: Bool) -> Promise<LokiMessage> {
/// Build a LokiMessage from a SignalMessage
///
/// - Parameters:
/// - signalMessage: the signal message
/// - timestamp: the original message timestamp (TSOutgoingMessage.timestamp)
/// - isPoWRequired: Should we calculate proof of work
/// - Returns: The loki message
public static func from(signalMessage: SignalMessage, timestamp: UInt64, requiringPoW isPoWRequired: Bool) -> Promise<LokiMessage> {
// To match the desktop application we have to take the data
// wrap it in an envelope, then
// wrap it in a websocket
return Promise<LokiMessage> { seal in
DispatchQueue.global(qos: .default).async {
guard let envelope = buildEnvelope(fromSignalMessage: signalMessage, timestamp: timestamp) else {
seal.reject(LokiAPI.Error.failedToWrapInEnvelope)
return
}
// Make the data
guard let websocket = wrapInWebsocket(envelope: envelope),
let serialized = try? websocket.serializedData() else {
seal.reject(LokiAPI.Error.failedToWrapInWebSocket)
return;
}
let data = serialized.base64EncodedString()
let destination = signalMessage["destination"] as! String
let data = signalMessage["content"] as! String
let ttl = LokiAPI.defaultMessageTTL
if isPoWRequired {
// timeIntervalSince1970 returns timestamp in seconds but the storage server only accepts timestamp in milliseconds
let timestamp = UInt64(Date().timeIntervalSince1970 * 1000)
if let nonce = ProofOfWork.calculate(data: data, pubKey: destination, timestamp: timestamp, ttl: ttl) {
let result = LokiMessage(destination: destination, data: data, ttl: ttl, timestamp: timestamp, nonce: nonce)
let now = UInt64(Date().timeIntervalSince1970 * 1000)
if let nonce = ProofOfWork.calculate(data: data, pubKey: destination, timestamp: now, ttl: ttl) {
let result = LokiMessage(destination: destination, data: data, ttl: ttl, timestamp: now, nonce: nonce)
seal.fulfill(result)
} else {
seal.reject(LokiAPI.Error.proofOfWorkCalculationFailed)
@ -45,6 +68,61 @@ public struct LokiMessage {
}
}
/// Wrap EnvelopeProto in a WebSocketProto
/// This is needed because it is done automatically on the desktop
private static func wrapInWebsocket(envelope: SSKProtoEnvelope) -> WebSocketProtoWebSocketMessage? {
do {
// This request is just a copy of the one on desktop
let requestBuilder = WebSocketProtoWebSocketRequestMessage.builder(verb: "PUT", path: "/api/v1/message", requestID: UInt64.random(in: 1..<UInt64.max))
let envelopeData = try envelope.serializedData()
requestBuilder.setBody(envelopeData)
// Build the websocket message
let builder = WebSocketProtoWebSocketMessage.builder(type: .request)
let request = try requestBuilder.build()
builder.setRequest(request)
return try builder.build()
} catch {
owsFailDebug("Loki Message: error building websocket message: \(error)")
return nil
}
}
/// Build the EnvelopeProto from SignalMessage
private static func buildEnvelope(fromSignalMessage signalMessage: SignalMessage, timestamp: UInt64) -> SSKProtoEnvelope? {
guard let ourKeys = SSKEnvironment.shared.identityManager.identityKeyPair() else {
owsFailDebug("error building envelope: identityManager.identityKeyPair() is invalid")
return nil;
}
do {
let ourPubKey = ourKeys.hexEncodedPublicKey
let params = ParamParser(dictionary: signalMessage)
let typeInt: Int32 = try params.required(key: "type")
guard let type: SSKProtoEnvelope.SSKProtoEnvelopeType = SSKProtoEnvelope.SSKProtoEnvelopeType(rawValue: typeInt) else {
Logger.error("`type` was invalid: \(typeInt)")
throw ParamParser.ParseError.invalidFormat("type")
}
let builder = SSKProtoEnvelope.builder(type: type, timestamp: timestamp)
builder.setSource(ourPubKey)
builder.setSourceDevice(OWSDevicePrimaryDeviceId)
if let content = try params.optionalBase64EncodedData(key: "content") {
builder.setContent(content)
}
return try builder.build()
} catch {
owsFailDebug("Loki Message: error building envelope: \(error)")
return nil
}
}
public func toJSON() -> JSON {
var result = [ "pubKey" : destination, "data" : data.description, "ttl" : String(ttl) ]
if let timestamp = timestamp, let nonce = nonce {

@ -1112,7 +1112,7 @@ NSString *const OWSMessageSenderRateLimitedException = @"RateLimitedException";
// Convert the message to a Loki message and send it using the Loki messaging API
NSDictionary *signalMessage = deviceMessages.firstObject;
BOOL isPoWRequired = YES; // TODO: Base on message type
[[LokiAPI objc_sendSignalMessage:signalMessage to:recipient.recipientId requiringPoW:isPoWRequired]
[[LokiAPI objc_sendSignalMessage:signalMessage to:recipient.recipientId timestamp:message.timestamp requiringPoW:isPoWRequired]
.thenOn(OWSDispatch.sendingQueue, ^(id result) {
[self messageSendDidSucceed:messageSend
deviceMessages:deviceMessages

Loading…
Cancel
Save