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.
241 lines
11 KiB
Swift
241 lines
11 KiB
Swift
// Copyright © 2023 Rangeproof Pty Ltd. All rights reserved.
|
|
|
|
import Foundation
|
|
import Combine
|
|
import GRDB
|
|
import SessionUtilitiesKit
|
|
import SessionSnodeKit
|
|
|
|
// MARK: - Log.Category
|
|
|
|
private extension Log.Category {
|
|
static let cat: Log.Category = .create("GroupLeavingJob", defaultLevel: .info)
|
|
}
|
|
|
|
// MARK: - GroupLeavingJob
|
|
|
|
public enum GroupLeavingJob: JobExecutor {
|
|
public static var maxFailureCount: Int = 0
|
|
public static var requiresThreadId: Bool = true
|
|
public static var requiresInteractionId: Bool = true
|
|
|
|
public static func run(
|
|
_ job: Job,
|
|
queue: DispatchQueue,
|
|
success: @escaping (Job, Bool) -> Void,
|
|
failure: @escaping (Job, Error, Bool) -> Void,
|
|
deferred: @escaping (Job) -> Void,
|
|
using dependencies: Dependencies
|
|
) {
|
|
guard
|
|
let detailsData: Data = job.details,
|
|
let details: Details = try? JSONDecoder(using: dependencies).decode(Details.self, from: detailsData),
|
|
let threadId: String = job.threadId,
|
|
let interactionId: Int64 = job.interactionId
|
|
else { return failure(job, JobRunnerError.missingRequiredDetails, true) }
|
|
|
|
let destination: Message.Destination = .closedGroup(groupPublicKey: threadId)
|
|
|
|
dependencies[singleton: .storage]
|
|
.writePublisher { db -> LeaveType in
|
|
guard
|
|
let threadVariant: SessionThread.Variant = try? SessionThread
|
|
.filter(id: threadId)
|
|
.select(.variant)
|
|
.asRequest(of: SessionThread.Variant.self)
|
|
.fetchOne(db)
|
|
else {
|
|
Log.error(.cat, "Failed due to non-existent group conversation")
|
|
throw MessageSenderError.noThread
|
|
}
|
|
guard (try? ClosedGroup.exists(db, id: threadId)) == true else {
|
|
Log.error(.cat, "Failed due to non-existent group")
|
|
throw MessageSenderError.invalidClosedGroupUpdate
|
|
}
|
|
|
|
let userSessionId: SessionId = dependencies[cache: .general].sessionId
|
|
let isAdminUser: Bool = GroupMember
|
|
.filter(GroupMember.Columns.groupId == threadId)
|
|
.filter(GroupMember.Columns.profileId == userSessionId.hexString)
|
|
.filter(GroupMember.Columns.role == GroupMember.Role.admin)
|
|
.isNotEmpty(db)
|
|
let numAdminUsers: Int = (try? GroupMember
|
|
.filter(GroupMember.Columns.groupId == threadId)
|
|
.filter(GroupMember.Columns.role == GroupMember.Role.admin)
|
|
.distinct()
|
|
.fetchCount(db))
|
|
.defaulting(to: 0)
|
|
|
|
switch (threadVariant, details.behaviour, (isAdminUser && numAdminUsers == 1)) {
|
|
case (.legacyGroup, _, _):
|
|
// Legacy group only supports the 'leave' behaviour so don't bother checking
|
|
return .leave(
|
|
try MessageSender.preparedSend(
|
|
db,
|
|
message: ClosedGroupControlMessage(kind: .memberLeft),
|
|
to: destination,
|
|
namespace: destination.defaultNamespace,
|
|
interactionId: job.interactionId,
|
|
fileIds: [],
|
|
using: dependencies
|
|
)
|
|
)
|
|
|
|
case (.group, .leave, false):
|
|
return .leave(
|
|
try SnodeAPI
|
|
.preparedBatch(
|
|
requests: [
|
|
try MessageSender.preparedSend(
|
|
db,
|
|
message: GroupUpdateMemberLeftMessage(),
|
|
to: destination,
|
|
namespace: destination.defaultNamespace,
|
|
interactionId: job.interactionId,
|
|
fileIds: [],
|
|
using: dependencies
|
|
),
|
|
try MessageSender.preparedSend(
|
|
db,
|
|
message: GroupUpdateMemberLeftNotificationMessage(),
|
|
to: destination,
|
|
namespace: destination.defaultNamespace,
|
|
interactionId: nil,
|
|
fileIds: [],
|
|
using: dependencies
|
|
)
|
|
],
|
|
requireAllBatchResponses: false,
|
|
swarmPublicKey: threadId,
|
|
using: dependencies
|
|
)
|
|
.map { _, _ in () }
|
|
)
|
|
|
|
case (.group, .delete, _), (.group, .leave, true):
|
|
try LibSession.deleteGroupForEveryone(
|
|
db,
|
|
groupSessionId: SessionId(.group, hex: threadId),
|
|
using: dependencies
|
|
)
|
|
|
|
return .delete
|
|
|
|
default: throw MessageSenderError.invalidClosedGroupUpdate
|
|
}
|
|
}
|
|
.flatMap { leaveType -> AnyPublisher<Void, Error> in
|
|
switch leaveType {
|
|
case .leave(let leaveMessage):
|
|
return leaveMessage
|
|
.send(using: dependencies)
|
|
.map { _ in () }
|
|
.eraseToAnyPublisher()
|
|
|
|
case .delete:
|
|
return ConfigurationSyncJob
|
|
.run(swarmPublicKey: threadId, using: dependencies)
|
|
.map { _ in () }
|
|
.eraseToAnyPublisher()
|
|
}
|
|
}
|
|
.tryCatch { error -> AnyPublisher<Void, Error> in
|
|
/// If it failed due to one of these errors then clear out any associated data (as the `SessionThread` exists but
|
|
/// either the data required to send the `MEMBER_LEFT` message doesn't or the user has had their access to the
|
|
/// group revoked which would leave the user in a state where they can't leave the group)
|
|
switch (error as? MessageSenderError, error as? SnodeAPIError) {
|
|
case (.invalidClosedGroupUpdate, _), (.noKeyPair, _), (.encryptionFailed, _),
|
|
(_, .unauthorised), (_, .invalidAuthentication):
|
|
return Just(()).setFailureType(to: Error.self).eraseToAnyPublisher()
|
|
|
|
default: throw error
|
|
}
|
|
}
|
|
.subscribe(on: queue, using: dependencies)
|
|
.receive(on: queue, using: dependencies)
|
|
.sinkUntilComplete(
|
|
receiveCompletion: { result in
|
|
switch result {
|
|
case .failure(let error):
|
|
// Update the interaction to indicate we failed to leave the group
|
|
dependencies[singleton: .storage].writeAsync { db in
|
|
let updatedBody: String = {
|
|
switch details.behaviour {
|
|
case .leave:
|
|
return "groupLeaveErrorFailed"
|
|
.put(key: "group_name", value: ((try? ClosedGroup.fetchOne(db, id: threadId))?.name ?? ""))
|
|
.localized()
|
|
case .delete:
|
|
return "groupLeaveErrorFailed"
|
|
.put(key: "group_name", value: ((try? ClosedGroup.fetchOne(db, id: threadId))?.name ?? ""))
|
|
.localized()
|
|
}
|
|
}()
|
|
|
|
try Interaction
|
|
.filter(id: interactionId)
|
|
.updateAll(
|
|
db,
|
|
Interaction.Columns.variant
|
|
.set(to: Interaction.Variant.infoGroupCurrentUserErrorLeaving),
|
|
Interaction.Columns.body.set(to: updatedBody)
|
|
)
|
|
}
|
|
|
|
failure(job, error, true)
|
|
|
|
case .finished:
|
|
// Remove all of the group data
|
|
dependencies[singleton: .storage].writeAsync { db in
|
|
try ClosedGroup.removeData(
|
|
db,
|
|
threadIds: [threadId],
|
|
dataToRemove: .allData,
|
|
calledFromConfig: nil,
|
|
using: dependencies
|
|
)
|
|
}
|
|
|
|
success(job, false)
|
|
}
|
|
}
|
|
)
|
|
}
|
|
}
|
|
|
|
// MARK: - GroupLeavingJob.Details
|
|
|
|
extension GroupLeavingJob {
|
|
public struct Details: Codable {
|
|
private enum CodingKeys: String, CodingKey {
|
|
case behaviour
|
|
}
|
|
|
|
public enum Behaviour: Int, Codable {
|
|
/// Will leave the group, deleting it from the current device but letting the group continue to exist as long
|
|
/// the current user isn't the only admin
|
|
case leave = 1
|
|
|
|
/// Will permanently delete the group, this will result in the group being deleted from all member devices
|
|
case delete = 2
|
|
}
|
|
|
|
public let behaviour: Behaviour
|
|
|
|
// MARK: - Initialization
|
|
|
|
public init(behaviour: Behaviour) {
|
|
self.behaviour = behaviour
|
|
}
|
|
}
|
|
}
|
|
|
|
// MARK: - Convenience
|
|
|
|
private extension GroupLeavingJob {
|
|
enum LeaveType {
|
|
case leave(Network.PreparedRequest<Void>)
|
|
case delete
|
|
}
|
|
}
|