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.
191 lines
7.9 KiB
Swift
191 lines
7.9 KiB
Swift
// Copyright © 2022 Rangeproof Pty Ltd. All rights reserved.
|
|
|
|
import Foundation
|
|
import Combine
|
|
import GRDB
|
|
import SessionUtilitiesKit
|
|
import SessionSnodeKit
|
|
|
|
extension MessageReceiver {
|
|
internal static func handleMessageRequestResponse(
|
|
_ db: Database,
|
|
message: MessageRequestResponse,
|
|
dependencies: SMKDependencies
|
|
) throws {
|
|
let userPublicKey = getUserHexEncodedPublicKey(db, dependencies: dependencies)
|
|
var blindedContactIds: [String] = []
|
|
|
|
// Ignore messages which were sent from the current user
|
|
guard
|
|
message.sender != userPublicKey,
|
|
let senderId: String = message.sender
|
|
else { throw MessageReceiverError.invalidMessage }
|
|
|
|
// Update profile if needed (want to do this regardless of whether the message exists or
|
|
// not to ensure the profile info gets sync between a users devices at every chance)
|
|
if let profile = message.profile {
|
|
let messageSentTimestamp: TimeInterval = (TimeInterval(message.sentTimestamp ?? 0) / 1000)
|
|
|
|
try ProfileManager.updateProfileIfNeeded(
|
|
db,
|
|
publicKey: senderId,
|
|
name: profile.displayName,
|
|
avatarUpdate: {
|
|
guard
|
|
let profilePictureUrl: String = profile.profilePictureUrl,
|
|
let profileKey: Data = profile.profileKey
|
|
else { return .none }
|
|
|
|
return .updateTo(
|
|
url: profilePictureUrl,
|
|
key: profileKey,
|
|
fileName: nil
|
|
)
|
|
}(),
|
|
sentTimestamp: messageSentTimestamp
|
|
)
|
|
}
|
|
|
|
// Prep the unblinded thread
|
|
let unblindedThread: SessionThread = try SessionThread
|
|
.fetchOrCreate(db, id: senderId, variant: .contact, shouldBeVisible: nil)
|
|
|
|
// Need to handle a `MessageRequestResponse` sent to a blinded thread (ie. check if the sender matches
|
|
// the blinded ids of any threads)
|
|
let blindedThreadIds: Set<String> = (try? SessionThread
|
|
.select(.id)
|
|
.filter(SessionThread.Columns.variant == SessionThread.Variant.contact)
|
|
.filter(
|
|
SessionThread.Columns.id.like("\(SessionId.Prefix.blinded15.rawValue)%") ||
|
|
SessionThread.Columns.id.like("\(SessionId.Prefix.blinded25.rawValue)%")
|
|
)
|
|
.asRequest(of: String.self)
|
|
.fetchSet(db))
|
|
.defaulting(to: [])
|
|
let pendingBlindedIdLookups: [BlindedIdLookup] = (try? BlindedIdLookup
|
|
.filter(blindedThreadIds.contains(BlindedIdLookup.Columns.blindedId))
|
|
.fetchAll(db))
|
|
.defaulting(to: [])
|
|
|
|
// Loop through all blinded threads and extract any interactions relating to the user accepting
|
|
// the message request
|
|
try pendingBlindedIdLookups.forEach { blindedIdLookup in
|
|
// If the sessionId matches the blindedId then this thread needs to be converted to an
|
|
// un-blinded thread
|
|
guard
|
|
dependencies.sodium.sessionId(
|
|
senderId,
|
|
matchesBlindedId: blindedIdLookup.blindedId,
|
|
serverPublicKey: blindedIdLookup.openGroupPublicKey,
|
|
genericHash: dependencies.genericHash
|
|
)
|
|
else { return }
|
|
|
|
// Update the lookup
|
|
_ = try blindedIdLookup
|
|
.with(sessionId: senderId)
|
|
.saved(db)
|
|
|
|
// Add the `blindedId` to an array so we can remove them at the end of processing
|
|
blindedContactIds.append(blindedIdLookup.blindedId)
|
|
|
|
// Update all interactions to be on the new thread
|
|
// Note: Pending `MessageSendJobs` _shouldn't_ be an issue as even if they are sent after the
|
|
// un-blinding of a thread, the logic when handling the sent messages should automatically
|
|
// assign them to the correct thread
|
|
try Interaction
|
|
.filter(Interaction.Columns.threadId == blindedIdLookup.blindedId)
|
|
.updateAll(db, Interaction.Columns.threadId.set(to: unblindedThread.id))
|
|
|
|
_ = try SessionThread
|
|
.deleteOrLeave(
|
|
db,
|
|
threadId: blindedIdLookup.blindedId,
|
|
threadVariant: .contact,
|
|
groupLeaveType: .forced,
|
|
calledFromConfigHandling: false
|
|
)
|
|
}
|
|
|
|
// Update the `didApproveMe` state of the sender
|
|
try updateContactApprovalStatusIfNeeded(
|
|
db,
|
|
senderSessionId: senderId,
|
|
threadId: nil
|
|
)
|
|
|
|
// If there were blinded contacts which have now been resolved to this contact then we should remove
|
|
// the blinded contact and we also need to assume that the 'sender' is a newly created contact and
|
|
// hence need to update it's `isApproved` state
|
|
if !blindedContactIds.isEmpty {
|
|
_ = try? Contact
|
|
.filter(ids: blindedContactIds)
|
|
.deleteAll(db)
|
|
|
|
try updateContactApprovalStatusIfNeeded(
|
|
db,
|
|
senderSessionId: userPublicKey,
|
|
threadId: unblindedThread.id
|
|
)
|
|
}
|
|
|
|
// Notify the user of their approval (Note: This will always appear in the un-blinded thread)
|
|
//
|
|
// Note: We want to do this last as it'll mean the un-blinded thread gets updated and the
|
|
// contact approval status will have been updated at this point (which will mean the
|
|
// `isMessageRequest` will return correctly after this is saved)
|
|
_ = try Interaction(
|
|
serverHash: message.serverHash,
|
|
threadId: unblindedThread.id,
|
|
authorId: senderId,
|
|
variant: .infoMessageRequestAccepted,
|
|
timestampMs: (
|
|
message.sentTimestamp.map { Int64($0) } ??
|
|
SnodeAPI.currentOffsetTimestampMs()
|
|
)
|
|
).inserted(db)
|
|
}
|
|
|
|
internal static func updateContactApprovalStatusIfNeeded(
|
|
_ db: Database,
|
|
senderSessionId: String,
|
|
threadId: String?
|
|
) throws {
|
|
let userPublicKey: String = getUserHexEncodedPublicKey(db)
|
|
|
|
// If the sender of the message was the current user
|
|
if senderSessionId == userPublicKey {
|
|
// Retrieve the contact for the thread the message was sent to (excluding 'NoteToSelf'
|
|
// threads) and if the contact isn't flagged as approved then do so
|
|
guard
|
|
let threadId: String = threadId,
|
|
let thread: SessionThread = try? SessionThread.fetchOne(db, id: threadId),
|
|
!thread.isNoteToSelf(db)
|
|
else { return }
|
|
|
|
// Sending a message to someone flags them as approved so create the contact record if
|
|
// it doesn't exist
|
|
let contact: Contact = Contact.fetchOrCreate(db, id: threadId)
|
|
|
|
guard !contact.isApproved else { return }
|
|
|
|
try? contact.save(db)
|
|
_ = try? Contact
|
|
.filter(id: threadId)
|
|
.updateAllAndConfig(db, Contact.Columns.isApproved.set(to: true))
|
|
}
|
|
else {
|
|
// The message was sent to the current user so flag their 'didApproveMe' as true (can't send a message to
|
|
// someone without approving them)
|
|
let contact: Contact = Contact.fetchOrCreate(db, id: senderSessionId)
|
|
|
|
guard !contact.didApproveMe else { return }
|
|
|
|
try? contact.save(db)
|
|
_ = try? Contact
|
|
.filter(id: senderSessionId)
|
|
.updateAllAndConfig(db, Contact.Columns.didApproveMe.set(to: true))
|
|
}
|
|
}
|
|
}
|