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.
106 lines
4.6 KiB
Swift
106 lines
4.6 KiB
Swift
// Copyright © 2022 Rangeproof Pty Ltd. All rights reserved.
|
|
|
|
import Foundation
|
|
import GRDB
|
|
import SessionSnodeKit
|
|
import SessionUtilitiesKit
|
|
|
|
extension MessageReceiver {
|
|
public static func handleUnsendRequest(
|
|
_ db: Database,
|
|
threadId: String,
|
|
threadVariant: SessionThread.Variant,
|
|
message: UnsendRequest,
|
|
using dependencies: Dependencies
|
|
) throws {
|
|
let userSessionId: SessionId = dependencies[cache: .general].sessionId
|
|
let senderIsLegacyGroupAdmin: Bool = {
|
|
switch (message.sender, threadVariant) {
|
|
case (.some(let sender), .legacyGroup):
|
|
return GroupMember
|
|
.filter(GroupMember.Columns.groupId == threadId)
|
|
.filter(GroupMember.Columns.profileId == sender)
|
|
.filter(GroupMember.Columns.role == GroupMember.Role.admin)
|
|
.isNotEmpty(db)
|
|
|
|
default: return false
|
|
}
|
|
}()
|
|
|
|
guard
|
|
senderIsLegacyGroupAdmin ||
|
|
message.sender == message.author ||
|
|
userSessionId.hexString == message.sender
|
|
else { return }
|
|
guard
|
|
let author: String = message.author,
|
|
let timestampMs: UInt64 = message.timestamp,
|
|
let interactionInfo: Interaction.ThreadInfo = try Interaction
|
|
.select(.id, .threadId)
|
|
.filter(Interaction.Columns.timestampMs == Int64(timestampMs))
|
|
.filter(Interaction.Columns.authorId == author)
|
|
.asRequest(of: Interaction.ThreadInfo.self)
|
|
.fetchOne(db)
|
|
else { return }
|
|
|
|
/// Retrieve the hashes which should be deleted first (these will be removed from the local
|
|
/// device in the `markAsDeleted` function) then call `markAsDeleted` to remove
|
|
/// message content
|
|
let hashes: Set<String> = try Interaction.serverHashesForDeletion(
|
|
db,
|
|
interactionIds: [interactionInfo.id]
|
|
)
|
|
try Interaction.markAsDeleted(
|
|
db,
|
|
threadId: threadId,
|
|
threadVariant: threadVariant,
|
|
interactionIds: [interactionInfo.id],
|
|
localOnly: false,
|
|
using: dependencies
|
|
)
|
|
|
|
/// If it's the `Note to Self` conversation then we want to just delete the interaction
|
|
if userSessionId.hexString == interactionInfo.threadId {
|
|
try Interaction.deleteOne(db, id: interactionInfo.id)
|
|
}
|
|
|
|
/// Can't delete from the legacy group swarm so only bother for contact conversations
|
|
switch threadVariant {
|
|
case .legacyGroup, .group, .community: break
|
|
case .contact:
|
|
dependencies[singleton: .storage]
|
|
.readPublisher { db in
|
|
try SnodeAPI.preparedDeleteMessages(
|
|
serverHashes: Array(hashes),
|
|
requireSuccessfulDeletion: false,
|
|
authMethod: try Authentication.with(
|
|
db,
|
|
swarmPublicKey: dependencies[cache: .general].sessionId.hexString,
|
|
using: dependencies
|
|
),
|
|
using: dependencies
|
|
)
|
|
}
|
|
.flatMap { $0.send(using: dependencies) }
|
|
.subscribe(on: DispatchQueue.global(qos: .background), using: dependencies)
|
|
.sinkUntilComplete(
|
|
receiveCompletion: { result in
|
|
switch result {
|
|
case .failure: break
|
|
case .finished:
|
|
/// Since the server deletion was successful we should also flag the `SnodeReceivedMessageInfo`
|
|
/// entries for the hashes as invalud (otherwise we might try to poll for a hash which no longer exists,
|
|
/// resulting in fetching the last 14 days of messages)
|
|
dependencies[singleton: .storage].writeAsync { db in
|
|
try SnodeReceivedMessageInfo.handlePotentialDeletedOrInvalidHash(
|
|
db,
|
|
potentiallyInvalidHashes: Array(hashes)
|
|
)
|
|
}
|
|
}
|
|
}
|
|
)
|
|
}
|
|
}
|
|
}
|