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.
617 lines
25 KiB
Swift
617 lines
25 KiB
Swift
3 years ago
|
// Copyright © 2022 Rangeproof Pty Ltd. All rights reserved.
|
||
|
|
||
|
import Foundation
|
||
3 years ago
|
import GRDB
|
||
3 years ago
|
import Sodium
|
||
3 years ago
|
import Curve25519Kit
|
||
4 years ago
|
import PromiseKit
|
||
3 years ago
|
import SessionUtilitiesKit
|
||
4 years ago
|
|
||
4 years ago
|
extension MessageSender {
|
||
3 years ago
|
public static var distributingKeyPairs: Atomic<[String: [ClosedGroupKeyPair]]> = Atomic([:])
|
||
4 years ago
|
|
||
3 years ago
|
public static func createClosedGroup(_ db: Database, name: String, members: Set<String>) throws -> Promise<SessionThread> {
|
||
|
let userPublicKey: String = getUserHexEncodedPublicKey()
|
||
|
var members: Set<String> = members
|
||
|
|
||
4 years ago
|
// Generate the group's public key
|
||
3 years ago
|
let groupPublicKey = Curve25519.generateKeyPair().hexEncodedPublicKey // Includes the 'SessionId.Prefix.standard' prefix
|
||
4 years ago
|
// Generate the key pair that'll be used for encryption and decryption
|
||
|
let encryptionKeyPair = Curve25519.generateKeyPair()
|
||
3 years ago
|
|
||
4 years ago
|
// Create the group
|
||
3 years ago
|
members.insert(userPublicKey) // Ensure the current user is included in the member list
|
||
|
let membersAsData = members.map { Data(hex: $0) }
|
||
4 years ago
|
let admins = [ userPublicKey ]
|
||
|
let adminsAsData = admins.map { Data(hex: $0) }
|
||
3 years ago
|
let formationTimestamp: TimeInterval = Date().timeIntervalSince1970
|
||
3 years ago
|
let thread: SessionThread = try SessionThread
|
||
|
.fetchOrCreate(db, id: groupPublicKey, variant: .closedGroup)
|
||
|
try ClosedGroup(
|
||
|
threadId: groupPublicKey,
|
||
|
name: name,
|
||
3 years ago
|
formationTimestamp: formationTimestamp
|
||
3 years ago
|
).insert(db)
|
||
|
|
||
|
try admins.forEach { adminId in
|
||
|
try GroupMember(
|
||
|
groupId: groupPublicKey,
|
||
|
profileId: adminId,
|
||
|
role: .admin
|
||
|
).insert(db)
|
||
|
}
|
||
|
|
||
4 years ago
|
// Send a closed group update message to all members individually
|
||
|
var promises: [Promise<Void>] = []
|
||
3 years ago
|
|
||
|
try members.forEach { adminId in
|
||
|
try GroupMember(
|
||
|
groupId: groupPublicKey,
|
||
|
profileId: adminId,
|
||
|
role: .admin
|
||
|
).insert(db)
|
||
|
}
|
||
|
|
||
|
try members.forEach { memberId in
|
||
|
let contactThread: SessionThread = try SessionThread
|
||
|
.fetchOrCreate(db, id: memberId, variant: .contact)
|
||
|
|
||
|
// Sending this non-durably is okay because we show a loader to the user. If they
|
||
|
// close the app while the loader is still showing, it's within expectation that
|
||
|
// the group creation might be incomplete.
|
||
|
promises.append(
|
||
|
try MessageSender.sendNonDurably(
|
||
|
db,
|
||
|
message: ClosedGroupControlMessage(
|
||
|
kind: .new(
|
||
|
publicKey: Data(hex: groupPublicKey),
|
||
|
name: name,
|
||
3 years ago
|
encryptionKeyPair: Box.KeyPair(
|
||
|
publicKey: encryptionKeyPair.publicKey.bytes,
|
||
|
secretKey: encryptionKeyPair.privateKey.bytes
|
||
|
),
|
||
3 years ago
|
members: membersAsData,
|
||
|
admins: adminsAsData,
|
||
|
expirationTimer: 0
|
||
3 years ago
|
),
|
||
3 years ago
|
// Note: We set this here to ensure the value matches the 'ClosedGroup'
|
||
|
// object we created
|
||
3 years ago
|
sentTimestampMs: UInt64(floor(formationTimestamp * 1000))
|
||
3 years ago
|
),
|
||
|
interactionId: nil,
|
||
|
in: contactThread
|
||
|
)
|
||
3 years ago
|
)
|
||
3 years ago
|
}
|
||
|
|
||
4 years ago
|
// Store the key pair
|
||
3 years ago
|
try ClosedGroupKeyPair(
|
||
3 years ago
|
threadId: groupPublicKey,
|
||
|
publicKey: encryptionKeyPair.publicKey,
|
||
|
secretKey: encryptionKeyPair.privateKey,
|
||
3 years ago
|
receivedTimestamp: Date().timeIntervalSince1970
|
||
|
).insert(db)
|
||
|
|
||
4 years ago
|
// Notify the PN server
|
||
3 years ago
|
promises.append(
|
||
|
PushNotificationAPI.performOperation(
|
||
|
.subscribe,
|
||
|
for: groupPublicKey,
|
||
|
publicKey: userPublicKey
|
||
|
)
|
||
|
)
|
||
|
|
||
4 years ago
|
// Notify the user
|
||
3 years ago
|
//
|
||
|
// Note: Intentionally don't want a 'serverHash' for closed group creation
|
||
|
_ = try Interaction(
|
||
|
threadId: thread.id,
|
||
|
authorId: userPublicKey,
|
||
|
variant: .infoClosedGroupCreated,
|
||
|
timestampMs: Int64(floor(Date().timeIntervalSince1970 * 1000))
|
||
|
).inserted(db)
|
||
|
|
||
4 years ago
|
// Start polling
|
||
|
ClosedGroupPoller.shared.startPolling(for: groupPublicKey)
|
||
3 years ago
|
|
||
4 years ago
|
return when(fulfilled: promises).map2 { thread }
|
||
|
}
|
||
4 years ago
|
|
||
3 years ago
|
/// Generates and distributes a new encryption key pair for the group with the given closed group. This sends an
|
||
|
/// `ENCRYPTION_KEY_PAIR` message to the group. The message contains a list of key pair wrappers. Each key
|
||
|
/// pair wrapper consists of the public key for which the wrapper is intended along with the newly generated key pair
|
||
4 years ago
|
/// encrypted for that public key.
|
||
|
///
|
||
|
/// The returned promise is fulfilled when the message has been sent to the group.
|
||
3 years ago
|
private static func generateAndSendNewEncryptionKeyPair(
|
||
|
_ db: Database,
|
||
|
targetMembers: Set<String>,
|
||
|
userPublicKey: String,
|
||
|
allGroupMembers: [GroupMember],
|
||
|
closedGroup: ClosedGroup,
|
||
|
thread: SessionThread
|
||
|
) throws -> Promise<Void> {
|
||
|
guard allGroupMembers.contains(where: { $0.role == .admin && $0.profileId == userPublicKey }) else {
|
||
|
return Promise(error: MessageSenderError.invalidClosedGroupUpdate)
|
||
4 years ago
|
}
|
||
|
// Generate the new encryption key pair
|
||
3 years ago
|
let legacyNewKeyPair: ECKeyPair = Curve25519.generateKeyPair()
|
||
|
let newKeyPair: ClosedGroupKeyPair = ClosedGroupKeyPair(
|
||
|
threadId: closedGroup.threadId,
|
||
|
publicKey: legacyNewKeyPair.publicKey,
|
||
|
secretKey: legacyNewKeyPair.privateKey,
|
||
|
receivedTimestamp: Date().timeIntervalSince1970
|
||
3 years ago
|
)
|
||
3 years ago
|
|
||
4 years ago
|
// Distribute it
|
||
3 years ago
|
let proto = try SNProtoKeyPair.builder(
|
||
|
publicKey: newKeyPair.publicKey,
|
||
|
privateKey: newKeyPair.secretKey
|
||
|
).build()
|
||
3 years ago
|
let plaintext = try proto.serializedData()
|
||
|
|
||
3 years ago
|
distributingKeyPairs.mutate {
|
||
|
$0[closedGroup.id] = ($0[closedGroup.id] ?? [])
|
||
|
.appending(newKeyPair)
|
||
|
}
|
||
3 years ago
|
|
||
|
do {
|
||
|
return try MessageSender
|
||
|
.sendNonDurably(
|
||
|
db,
|
||
|
message: ClosedGroupControlMessage(
|
||
|
kind: .encryptionKeyPair(
|
||
|
publicKey: nil,
|
||
|
wrappers: targetMembers.map { memberPublicKey in
|
||
|
ClosedGroupControlMessage.KeyPairWrapper(
|
||
|
publicKey: memberPublicKey,
|
||
|
encryptedKeyPair: try MessageSender.encryptWithSessionProtocol(
|
||
|
plaintext,
|
||
|
for: memberPublicKey
|
||
|
)
|
||
|
)
|
||
|
}
|
||
|
)
|
||
|
),
|
||
|
interactionId: nil,
|
||
|
in: thread
|
||
|
)
|
||
|
.done {
|
||
|
/// Store it **after** having sent out the message to the group
|
||
|
GRDBStorage.shared.write { db in
|
||
3 years ago
|
try newKeyPair.insert(db)
|
||
3 years ago
|
|
||
3 years ago
|
distributingKeyPairs.mutate {
|
||
|
if let index = ($0[closedGroup.id] ?? []).firstIndex(of: newKeyPair) {
|
||
|
$0[closedGroup.id] = ($0[closedGroup.id] ?? [])
|
||
|
.removing(index: index)
|
||
|
}
|
||
3 years ago
|
}
|
||
|
}
|
||
|
}
|
||
|
.map { _ in }
|
||
|
}
|
||
|
catch {
|
||
|
return Promise(error: MessageSenderError.invalidClosedGroupUpdate)
|
||
|
}
|
||
4 years ago
|
}
|
||
4 years ago
|
|
||
3 years ago
|
public static func update(
|
||
|
_ db: Database,
|
||
|
groupPublicKey: String,
|
||
|
with members: Set<String>,
|
||
|
name: String
|
||
|
) throws -> Promise<Void> {
|
||
4 years ago
|
// Get the group, check preconditions & prepare
|
||
3 years ago
|
guard let thread: SessionThread = try? SessionThread.fetchOne(db, id: groupPublicKey) else {
|
||
4 years ago
|
SNLog("Can't update nonexistent closed group.")
|
||
3 years ago
|
return Promise(error: MessageSenderError.noThread)
|
||
4 years ago
|
}
|
||
3 years ago
|
guard let closedGroup: ClosedGroup = try? thread.closedGroup.fetchOne(db) else {
|
||
|
return Promise(error: MessageSenderError.invalidClosedGroupUpdate)
|
||
|
}
|
||
|
|
||
|
let userPublicKey: String = getUserHexEncodedPublicKey(db)
|
||
|
|
||
4 years ago
|
// Update name if needed
|
||
3 years ago
|
if name != closedGroup.name {
|
||
|
// Update the group
|
||
|
let updatedClosedGroup: ClosedGroup = closedGroup.with(name: name)
|
||
|
try updatedClosedGroup.save(db)
|
||
|
|
||
|
// Notify the user
|
||
|
let interaction: Interaction = try Interaction(
|
||
|
threadId: thread.id,
|
||
|
authorId: userPublicKey,
|
||
|
variant: .infoClosedGroupUpdated,
|
||
|
body: ClosedGroupControlMessage.Kind
|
||
|
.nameChange(name: name)
|
||
|
.infoMessage(db, sender: userPublicKey),
|
||
|
timestampMs: Int64(floor(Date().timeIntervalSince1970 * 1000))
|
||
|
).inserted(db)
|
||
|
|
||
3 years ago
|
guard let interactionId: Int64 = interaction.id else { throw StorageError.objectNotSaved }
|
||
3 years ago
|
|
||
|
// Send the update to the group
|
||
|
let closedGroupControlMessage = ClosedGroupControlMessage(kind: .nameChange(name: name))
|
||
|
try MessageSender.send(
|
||
|
db,
|
||
|
message: closedGroupControlMessage,
|
||
|
interactionId: interactionId,
|
||
|
in: thread
|
||
|
)
|
||
|
}
|
||
|
|
||
|
// Retrieve member info
|
||
|
guard let allGroupMembers: [GroupMember] = try? closedGroup.allMembers.fetchAll(db) else {
|
||
|
return Promise(error: MessageSenderError.invalidClosedGroupUpdate)
|
||
|
}
|
||
|
|
||
|
let standardAndZombieMemberIds: [String] = allGroupMembers
|
||
|
.filter { $0.role == .standard || $0.role == .zombie }
|
||
|
.map { $0.profileId }
|
||
|
let addedMembers: Set<String> = members.subtracting(standardAndZombieMemberIds)
|
||
|
|
||
4 years ago
|
// Add members if needed
|
||
3 years ago
|
if !addedMembers.isEmpty {
|
||
|
do {
|
||
|
try addMembers(
|
||
|
db,
|
||
|
addedMembers: addedMembers,
|
||
|
userPublicKey: userPublicKey,
|
||
|
allGroupMembers: allGroupMembers,
|
||
|
closedGroup: closedGroup,
|
||
|
thread: thread
|
||
|
)
|
||
|
}
|
||
|
catch {
|
||
|
return Promise(error: MessageSenderError.invalidClosedGroupUpdate)
|
||
|
}
|
||
4 years ago
|
}
|
||
3 years ago
|
|
||
|
// Remove members if needed
|
||
|
let removedMembers: Set<String> = Set(standardAndZombieMemberIds).subtracting(members)
|
||
|
|
||
|
if !removedMembers.isEmpty {
|
||
|
do {
|
||
|
return try removeMembers(
|
||
|
db,
|
||
|
removedMembers: removedMembers,
|
||
|
userPublicKey: userPublicKey,
|
||
|
allGroupMembers: allGroupMembers,
|
||
|
closedGroup: closedGroup,
|
||
|
thread: thread
|
||
|
)
|
||
|
}
|
||
|
catch {
|
||
|
return Promise(error: MessageSenderError.invalidClosedGroupUpdate)
|
||
|
}
|
||
4 years ago
|
}
|
||
3 years ago
|
|
||
4 years ago
|
return Promise.value(())
|
||
4 years ago
|
}
|
||
|
|
||
3 years ago
|
|
||
|
/// Adds `newMembers` to the group with the given closed group. This sends a `MEMBERS_ADDED` message to the group, and a
|
||
4 years ago
|
/// `NEW` message to the members that were added (using one-on-one channels).
|
||
3 years ago
|
private static func addMembers(
|
||
|
_ db: Database,
|
||
|
addedMembers: Set<String>,
|
||
|
userPublicKey: String,
|
||
|
allGroupMembers: [GroupMember],
|
||
|
closedGroup: ClosedGroup,
|
||
|
thread: SessionThread
|
||
|
) throws {
|
||
|
guard let disappearingMessagesConfig: DisappearingMessagesConfiguration = try thread.disappearingMessagesConfiguration.fetchOne(db) else {
|
||
3 years ago
|
throw StorageError.objectNotFound
|
||
4 years ago
|
}
|
||
3 years ago
|
guard let encryptionKeyPair: ClosedGroupKeyPair = try closedGroup.fetchLatestKeyPair(db) else {
|
||
3 years ago
|
throw StorageError.objectNotFound
|
||
4 years ago
|
}
|
||
3 years ago
|
|
||
|
let groupMemberIds: [String] = allGroupMembers
|
||
|
.filter { $0.role == .standard }
|
||
|
.map { $0.profileId }
|
||
|
let groupAdminIds: [String] = allGroupMembers
|
||
|
.filter { $0.role == .admin }
|
||
|
.map { $0.profileId }
|
||
|
let members: Set<String> = Set(groupMemberIds).union(addedMembers)
|
||
|
let membersAsData: [Data] = members.map { Data(hex: $0) }
|
||
|
let adminsAsData: [Data] = groupAdminIds.map { Data(hex: $0) }
|
||
|
|
||
|
// Notify the user
|
||
|
let interaction: Interaction = try Interaction(
|
||
|
threadId: thread.id,
|
||
|
authorId: userPublicKey,
|
||
|
variant: .infoClosedGroupUpdated,
|
||
|
body: ClosedGroupControlMessage.Kind
|
||
|
.membersAdded(members: addedMembers.map { Data(hex: $0) })
|
||
|
.infoMessage(db, sender: userPublicKey),
|
||
|
timestampMs: Int64(floor(Date().timeIntervalSince1970 * 1000))
|
||
|
).inserted(db)
|
||
|
|
||
3 years ago
|
guard let interactionId: Int64 = interaction.id else { throw StorageError.objectNotSaved }
|
||
3 years ago
|
|
||
4 years ago
|
// Send the update to the group
|
||
3 years ago
|
try MessageSender.send(
|
||
|
db,
|
||
|
message: ClosedGroupControlMessage(
|
||
|
kind: .membersAdded(members: addedMembers.map { Data(hex: $0) })
|
||
|
),
|
||
|
interactionId: interactionId,
|
||
|
in: thread
|
||
|
)
|
||
|
|
||
|
try addedMembers.forEach { member in
|
||
|
// Send updates to the new members individually
|
||
|
let thread: SessionThread = try SessionThread
|
||
|
.fetchOrCreate(db, id: member, variant: .contact)
|
||
|
|
||
|
try MessageSender.send(
|
||
|
db,
|
||
|
message: ClosedGroupControlMessage(
|
||
|
kind: .new(
|
||
|
publicKey: Data(hex: closedGroup.id),
|
||
|
name: closedGroup.name,
|
||
|
encryptionKeyPair: Box.KeyPair(
|
||
|
publicKey: encryptionKeyPair.publicKey.bytes,
|
||
|
secretKey: encryptionKeyPair.secretKey.bytes
|
||
|
),
|
||
|
members: membersAsData,
|
||
|
admins: adminsAsData,
|
||
|
expirationTimer: (disappearingMessagesConfig.isEnabled ?
|
||
|
UInt32(floor(disappearingMessagesConfig.durationSeconds)) :
|
||
|
0
|
||
|
)
|
||
|
)
|
||
|
),
|
||
|
interactionId: nil,
|
||
|
in: thread
|
||
|
)
|
||
|
|
||
|
// Add the users to the group
|
||
|
try GroupMember(
|
||
|
groupId: closedGroup.id,
|
||
|
profileId: member,
|
||
|
role: .standard
|
||
|
).insert(db)
|
||
4 years ago
|
}
|
||
4 years ago
|
}
|
||
3 years ago
|
|
||
4 years ago
|
/// Removes `membersToRemove` from the group with the given `groupPublicKey`. Only the admin can remove members, and when they do
|
||
|
/// they generate and distribute a new encryption key pair for the group. A member cannot leave a group using this method. For that they should use
|
||
|
/// `leave(:using:)`.
|
||
|
///
|
||
|
/// The returned promise is fulfilled when the `MEMBERS_REMOVED` message has been sent to the group AND the new encryption key pair has been
|
||
|
/// generated and distributed.
|
||
3 years ago
|
private static func removeMembers(
|
||
|
_ db: Database,
|
||
|
removedMembers: Set<String>,
|
||
|
userPublicKey: String,
|
||
|
allGroupMembers: [GroupMember],
|
||
|
closedGroup: ClosedGroup,
|
||
|
thread: SessionThread
|
||
|
) throws -> Promise<Void> {
|
||
|
guard !removedMembers.contains(userPublicKey) else {
|
||
4 years ago
|
SNLog("Invalid closed group update.")
|
||
3 years ago
|
throw MessageSenderError.invalidClosedGroupUpdate
|
||
4 years ago
|
}
|
||
3 years ago
|
guard allGroupMembers.contains(where: { $0.role == .admin && $0.profileId == userPublicKey }) else {
|
||
4 years ago
|
SNLog("Only an admin can remove members from a group.")
|
||
3 years ago
|
throw MessageSenderError.invalidClosedGroupUpdate
|
||
4 years ago
|
}
|
||
3 years ago
|
|
||
|
let groupMemberIds: [String] = allGroupMembers
|
||
|
.filter { $0.role == .standard }
|
||
|
.map { $0.profileId }
|
||
|
let groupZombieIds: [String] = allGroupMembers
|
||
|
.filter { $0.role == .zombie }
|
||
|
.map { $0.profileId }
|
||
|
let members: Set<String> = Set(groupMemberIds).subtracting(removedMembers)
|
||
|
|
||
|
// Update zombie * member list
|
||
|
try allGroupMembers
|
||
|
.filter { member in
|
||
|
removedMembers.contains(member.profileId) && (
|
||
|
member.role == .standard ||
|
||
|
member.role == .zombie
|
||
|
)
|
||
|
}
|
||
|
.forEach { try $0.delete(db) }
|
||
|
|
||
|
let interactionId: Int64?
|
||
|
|
||
4 years ago
|
// Notify the user if needed (not if only zombie members were removed)
|
||
3 years ago
|
if !removedMembers.subtracting(groupZombieIds).isEmpty {
|
||
|
let interaction: Interaction = try Interaction(
|
||
|
threadId: thread.id,
|
||
|
authorId: userPublicKey,
|
||
|
variant: .infoClosedGroupUpdated,
|
||
|
body: ClosedGroupControlMessage.Kind
|
||
|
.membersRemoved(members: removedMembers.map { Data(hex: $0) })
|
||
|
.infoMessage(db, sender: userPublicKey),
|
||
|
timestampMs: Int64(floor(Date().timeIntervalSince1970 * 1000))
|
||
|
).inserted(db)
|
||
|
|
||
3 years ago
|
guard let newInteractionId: Int64 = interaction.id else { throw StorageError.objectNotSaved }
|
||
3 years ago
|
|
||
|
interactionId = newInteractionId
|
||
4 years ago
|
}
|
||
3 years ago
|
else {
|
||
|
interactionId = nil
|
||
|
}
|
||
|
|
||
|
// Send the update to the group and generate + distribute a new encryption key pair
|
||
|
let promise = try MessageSender
|
||
|
.sendNonDurably(
|
||
|
db,
|
||
|
message: ClosedGroupControlMessage(
|
||
|
kind: .membersRemoved(
|
||
|
members: removedMembers.map { Data(hex: $0) }
|
||
|
)
|
||
|
),
|
||
|
interactionId: interactionId,
|
||
|
in: thread
|
||
|
)
|
||
|
.map { _ in
|
||
|
try generateAndSendNewEncryptionKeyPair(
|
||
|
db,
|
||
|
targetMembers: members,
|
||
|
userPublicKey: userPublicKey,
|
||
|
allGroupMembers: allGroupMembers,
|
||
|
closedGroup: closedGroup,
|
||
|
thread: thread
|
||
|
)
|
||
|
}
|
||
|
.map { _ in }
|
||
|
|
||
4 years ago
|
return promise
|
||
|
}
|
||
|
|
||
3 years ago
|
/// Leave the group with the given `groupPublicKey`. If the current user is the admin, the group is disbanded entirely. If the
|
||
|
/// user is a regular member they'll be marked as a "zombie" member by the other users in the group (upon receiving the leave
|
||
|
/// message). The admin can then truly remove them later.
|
||
4 years ago
|
///
|
||
3 years ago
|
/// This function also removes all encryption key pairs associated with the closed group and the group's public key, and
|
||
|
/// unregisters from push notifications.
|
||
4 years ago
|
///
|
||
|
/// The returned promise is fulfilled when the `MEMBER_LEFT` message has been sent to the group.
|
||
3 years ago
|
public static func leave(_ db: Database, groupPublicKey: String) throws -> Promise<Void> {
|
||
|
guard let thread: SessionThread = try? SessionThread.fetchOne(db, id: groupPublicKey) else {
|
||
4 years ago
|
SNLog("Can't leave nonexistent closed group.")
|
||
3 years ago
|
return Promise(error: MessageSenderError.noThread)
|
||
4 years ago
|
}
|
||
3 years ago
|
guard let closedGroup: ClosedGroup = try? thread.closedGroup.fetchOne(db) else {
|
||
|
return Promise(error: MessageSenderError.invalidClosedGroupUpdate)
|
||
|
}
|
||
|
|
||
|
let userPublicKey: String = getUserHexEncodedPublicKey(db)
|
||
|
|
||
|
// Notify the user
|
||
|
let interaction: Interaction = try Interaction(
|
||
|
threadId: thread.id,
|
||
|
authorId: userPublicKey,
|
||
|
variant: .infoClosedGroupCurrentUserLeft,
|
||
|
body: ClosedGroupControlMessage.Kind
|
||
|
.memberLeft
|
||
|
.infoMessage(db, sender: userPublicKey),
|
||
|
timestampMs: Int64(floor(Date().timeIntervalSince1970 * 1000))
|
||
|
).inserted(db)
|
||
|
|
||
|
guard let interactionId: Int64 = interaction.id else {
|
||
3 years ago
|
throw StorageError.objectNotSaved
|
||
3 years ago
|
}
|
||
|
|
||
4 years ago
|
// Send the update to the group
|
||
3 years ago
|
let promise = try MessageSender
|
||
|
.sendNonDurably(
|
||
|
db,
|
||
|
message: ClosedGroupControlMessage(
|
||
|
kind: .memberLeft
|
||
|
),
|
||
|
interactionId: interactionId,
|
||
|
in: thread
|
||
|
)
|
||
|
.done {
|
||
3 years ago
|
// Remove the group from the database and unsubscribe from PNs
|
||
|
ClosedGroupPoller.shared.stopPolling(for: groupPublicKey)
|
||
|
|
||
3 years ago
|
GRDBStorage.shared.write { db in
|
||
|
_ = try closedGroup
|
||
|
.keyPairs
|
||
|
.deleteAll(db)
|
||
|
|
||
|
let _ = PushNotificationAPI.performOperation(
|
||
|
.unsubscribe,
|
||
|
for: groupPublicKey,
|
||
|
publicKey: userPublicKey
|
||
|
)
|
||
|
}
|
||
4 years ago
|
}
|
||
3 years ago
|
.map { _ in }
|
||
|
|
||
4 years ago
|
// Update the group
|
||
3 years ago
|
_ = try closedGroup
|
||
|
.allMembers
|
||
|
.deleteAll(db)
|
||
3 years ago
|
|
||
4 years ago
|
// Return
|
||
|
return promise
|
||
4 years ago
|
}
|
||
|
|
||
4 years ago
|
/*
|
||
4 years ago
|
public static func requestEncryptionKeyPair(for groupPublicKey: String, using transaction: YapDatabaseReadWriteTransaction) throws {
|
||
4 years ago
|
#if DEBUG
|
||
|
preconditionFailure("Shouldn't currently be in use.")
|
||
|
#endif
|
||
4 years ago
|
// Get the group, check preconditions & prepare
|
||
|
let groupID = LKGroupUtilities.getEncodedClosedGroupIDAsData(groupPublicKey)
|
||
|
let threadID = TSGroupThread.threadId(fromGroupId: groupID)
|
||
|
guard let thread = TSGroupThread.fetch(uniqueId: threadID, transaction: transaction) else {
|
||
|
SNLog("Can't request encryption key pair for nonexistent closed group.")
|
||
|
throw Error.noThread
|
||
|
}
|
||
|
let group = thread.groupModel
|
||
|
guard group.groupMemberIds.contains(getUserHexEncodedPublicKey()) else { return }
|
||
|
// Send the request to the group
|
||
|
let closedGroupControlMessage = ClosedGroupControlMessage(kind: .encryptionKeyPairRequest)
|
||
|
MessageSender.send(closedGroupControlMessage, in: thread, using: transaction)
|
||
|
}
|
||
4 years ago
|
*/
|
||
4 years ago
|
|
||
3 years ago
|
public static func sendLatestEncryptionKeyPair(_ db: Database, to publicKey: String, for groupPublicKey: String) {
|
||
|
guard let thread: SessionThread = try? SessionThread.fetchOne(db, id: groupPublicKey) else {
|
||
4 years ago
|
return SNLog("Couldn't send key pair for nonexistent closed group.")
|
||
4 years ago
|
}
|
||
3 years ago
|
guard let closedGroup: ClosedGroup = try? thread.closedGroup.fetchOne(db) else {
|
||
|
return
|
||
|
}
|
||
|
guard let allGroupMembers: [GroupMember] = try? closedGroup.allMembers.fetchAll(db) else {
|
||
|
return
|
||
|
}
|
||
|
guard allGroupMembers.contains(where: { $0.role == .standard && $0.profileId == publicKey }) else {
|
||
4 years ago
|
return SNLog("Refusing to send latest encryption key pair to non-member.")
|
||
4 years ago
|
}
|
||
3 years ago
|
|
||
4 years ago
|
// Get the latest encryption key pair
|
||
3 years ago
|
var maybeKeyPair: ClosedGroupKeyPair? = distributingKeyPairs.wrappedValue[groupPublicKey]?.last
|
||
3 years ago
|
|
||
3 years ago
|
if maybeKeyPair == nil {
|
||
|
maybeKeyPair = try? closedGroup.fetchLatestKeyPair(db)
|
||
3 years ago
|
}
|
||
|
|
||
3 years ago
|
guard let keyPair: ClosedGroupKeyPair = maybeKeyPair else { return }
|
||
3 years ago
|
|
||
4 years ago
|
// Send it
|
||
3 years ago
|
do {
|
||
|
let proto = try SNProtoKeyPair.builder(
|
||
3 years ago
|
publicKey: keyPair.publicKey,
|
||
|
privateKey: keyPair.secretKey
|
||
3 years ago
|
).build()
|
||
|
let plaintext = try proto.serializedData()
|
||
|
let thread: SessionThread = try SessionThread
|
||
|
.fetchOrCreate(db, id: publicKey, variant: .contact)
|
||
|
let ciphertext = try MessageSender.encryptWithSessionProtocol(plaintext, for: publicKey)
|
||
|
|
||
|
SNLog("Sending latest encryption key pair to: \(publicKey).")
|
||
|
try MessageSender.send(
|
||
|
db,
|
||
|
message: ClosedGroupControlMessage(
|
||
|
kind: .encryptionKeyPair(
|
||
|
publicKey: Data(hex: groupPublicKey),
|
||
|
wrappers: [
|
||
|
ClosedGroupControlMessage.KeyPairWrapper(
|
||
|
publicKey: publicKey,
|
||
|
encryptedKeyPair: ciphertext
|
||
|
)
|
||
|
]
|
||
|
)
|
||
|
),
|
||
|
interactionId: nil,
|
||
|
in: thread
|
||
|
)
|
||
|
}
|
||
|
catch {}
|
||
4 years ago
|
}
|
||
4 years ago
|
}
|