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.
273 lines
13 KiB
Swift
273 lines
13 KiB
Swift
4 years ago
|
import SessionProtocolKit
|
||
|
import SessionUtilitiesKit
|
||
|
|
||
4 years ago
|
public final class ClosedGroupControlMessage : ControlMessage {
|
||
4 years ago
|
public var kind: Kind?
|
||
|
|
||
4 years ago
|
public override var ttl: UInt64 {
|
||
|
switch kind {
|
||
|
case .encryptionKeyPair: return 4 * 24 * 60 * 60 * 1000
|
||
|
default: return 2 * 24 * 60 * 60 * 1000
|
||
|
}
|
||
|
}
|
||
|
|
||
4 years ago
|
// MARK: Kind
|
||
|
public enum Kind : CustomStringConvertible {
|
||
|
case new(publicKey: Data, name: String, encryptionKeyPair: ECKeyPair, members: [Data], admins: [Data])
|
||
4 years ago
|
/// - Note: Deprecated in favor of more explicit group updates.
|
||
4 years ago
|
case update(name: String, members: [Data])
|
||
|
case encryptionKeyPair([KeyPairWrapper]) // The new encryption key pair encrypted for each member individually
|
||
4 years ago
|
case nameChange(name: String)
|
||
|
case usersAdded(members: [Data])
|
||
|
case usersRemoved(members: [Data])
|
||
|
case userLeft
|
||
4 years ago
|
|
||
|
public var description: String {
|
||
|
switch self {
|
||
|
case .new: return "new"
|
||
|
case .update: return "update"
|
||
|
case .encryptionKeyPair: return "encryptionKeyPair"
|
||
4 years ago
|
case .nameChange: return "nameChange"
|
||
|
case .usersAdded: return "usersAdded"
|
||
|
case .usersRemoved: return "usersRemoved"
|
||
|
case .userLeft: return "userLeft"
|
||
4 years ago
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// MARK: Key Pair Wrapper
|
||
|
@objc(SNKeyPairWrapper)
|
||
|
public final class KeyPairWrapper : NSObject, NSCoding { // NSObject/NSCoding conformance is needed for YapDatabase compatibility
|
||
|
public var publicKey: String?
|
||
|
public var encryptedKeyPair: Data?
|
||
|
|
||
|
public var isValid: Bool { publicKey != nil && encryptedKeyPair != nil }
|
||
|
|
||
|
public init(publicKey: String, encryptedKeyPair: Data) {
|
||
|
self.publicKey = publicKey
|
||
|
self.encryptedKeyPair = encryptedKeyPair
|
||
|
}
|
||
|
|
||
|
public required init?(coder: NSCoder) {
|
||
|
if let publicKey = coder.decodeObject(forKey: "publicKey") as! String? { self.publicKey = publicKey }
|
||
|
if let encryptedKeyPair = coder.decodeObject(forKey: "encryptedKeyPair") as! Data? { self.encryptedKeyPair = encryptedKeyPair }
|
||
|
}
|
||
|
|
||
|
public func encode(with coder: NSCoder) {
|
||
|
coder.encode(publicKey, forKey: "publicKey")
|
||
|
coder.encode(encryptedKeyPair, forKey: "encryptedKeyPair")
|
||
|
}
|
||
|
|
||
4 years ago
|
public static func fromProto(_ proto: SNProtoDataMessageClosedGroupControlMessageKeyPairWrapper) -> KeyPairWrapper? {
|
||
4 years ago
|
return KeyPairWrapper(publicKey: proto.publicKey.toHexString(), encryptedKeyPair: proto.encryptedKeyPair)
|
||
|
}
|
||
|
|
||
4 years ago
|
public func toProto() -> SNProtoDataMessageClosedGroupControlMessageKeyPairWrapper? {
|
||
4 years ago
|
guard let publicKey = publicKey, let encryptedKeyPair = encryptedKeyPair else { return nil }
|
||
4 years ago
|
let result = SNProtoDataMessageClosedGroupControlMessageKeyPairWrapper.builder(publicKey: Data(hex: publicKey), encryptedKeyPair: encryptedKeyPair)
|
||
4 years ago
|
do {
|
||
|
return try result.build()
|
||
|
} catch {
|
||
|
SNLog("Couldn't construct key pair wrapper proto from: \(self).")
|
||
|
return nil
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// MARK: Initialization
|
||
|
public override init() { super.init() }
|
||
|
|
||
|
internal init(kind: Kind) {
|
||
|
super.init()
|
||
|
self.kind = kind
|
||
|
}
|
||
|
|
||
|
// MARK: Validation
|
||
|
public override var isValid: Bool {
|
||
|
guard super.isValid, let kind = kind else { return false }
|
||
|
switch kind {
|
||
|
case .new(let publicKey, let name, let encryptionKeyPair, let members, let admins):
|
||
|
return !publicKey.isEmpty && !name.isEmpty && !encryptionKeyPair.publicKey.isEmpty
|
||
|
&& !encryptionKeyPair.privateKey.isEmpty && !members.isEmpty && !admins.isEmpty
|
||
4 years ago
|
case .update(let name, _):
|
||
|
return !name.isEmpty
|
||
4 years ago
|
case .encryptionKeyPair: return true
|
||
4 years ago
|
case .nameChange(let name): return !name.isEmpty
|
||
|
case .usersAdded(let members): return !members.isEmpty
|
||
|
case .usersRemoved(let members): return !members.isEmpty
|
||
|
case .userLeft: return true
|
||
4 years ago
|
}
|
||
|
}
|
||
|
|
||
|
// MARK: Coding
|
||
|
public required init?(coder: NSCoder) {
|
||
|
super.init(coder: coder)
|
||
|
guard let rawKind = coder.decodeObject(forKey: "kind") as? String else { return nil }
|
||
|
switch rawKind {
|
||
|
case "new":
|
||
|
guard let publicKey = coder.decodeObject(forKey: "publicKey") as? Data,
|
||
|
let name = coder.decodeObject(forKey: "name") as? String,
|
||
|
let encryptionKeyPair = coder.decodeObject(forKey: "encryptionKeyPair") as? ECKeyPair,
|
||
|
let members = coder.decodeObject(forKey: "members") as? [Data],
|
||
|
let admins = coder.decodeObject(forKey: "admins") as? [Data] else { return nil }
|
||
|
self.kind = .new(publicKey: publicKey, name: name, encryptionKeyPair: encryptionKeyPair, members: members, admins: admins)
|
||
|
case "update":
|
||
|
guard let name = coder.decodeObject(forKey: "name") as? String,
|
||
|
let members = coder.decodeObject(forKey: "members") as? [Data] else { return nil }
|
||
|
self.kind = .update(name: name, members: members)
|
||
|
case "encryptionKeyPair":
|
||
|
guard let wrappers = coder.decodeObject(forKey: "wrappers") as? [KeyPairWrapper] else { return nil }
|
||
|
self.kind = .encryptionKeyPair(wrappers)
|
||
4 years ago
|
case "nameChange":
|
||
|
guard let name = coder.decodeObject(forKey: "name") as? String else { return nil }
|
||
|
self.kind = .nameChange(name: name)
|
||
|
case "usersAdded":
|
||
|
guard let members = coder.decodeObject(forKey: "members") as? [Data] else { return nil }
|
||
|
self.kind = .usersAdded(members: members)
|
||
|
case "usersRemoved":
|
||
|
guard let members = coder.decodeObject(forKey: "members") as? [Data] else { return nil }
|
||
|
self.kind = .usersRemoved(members: members)
|
||
|
case "userLeft":
|
||
|
self.kind = .userLeft
|
||
4 years ago
|
default: return nil
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public override func encode(with coder: NSCoder) {
|
||
|
super.encode(with: coder)
|
||
|
guard let kind = kind else { return }
|
||
|
switch kind {
|
||
|
case .new(let publicKey, let name, let encryptionKeyPair, let members, let admins):
|
||
|
coder.encode("new", forKey: "kind")
|
||
|
coder.encode(publicKey, forKey: "publicKey")
|
||
|
coder.encode(name, forKey: "name")
|
||
|
coder.encode(encryptionKeyPair, forKey: "encryptionKeyPair")
|
||
|
coder.encode(members, forKey: "members")
|
||
|
coder.encode(admins, forKey: "admins")
|
||
|
case .update(let name, let members):
|
||
|
coder.encode("update", forKey: "kind")
|
||
|
coder.encode(name, forKey: "name")
|
||
|
coder.encode(members, forKey: "members")
|
||
|
case .encryptionKeyPair(let wrappers):
|
||
|
coder.encode("encryptionKeyPair", forKey: "kind")
|
||
|
coder.encode(wrappers, forKey: "wrappers")
|
||
4 years ago
|
case .nameChange(let name):
|
||
|
coder.encode("nameChange", forKey: "kind")
|
||
|
coder.encode(name, forKey: "name")
|
||
|
case .usersAdded(let members):
|
||
|
coder.encode("usersAdded", forKey: "kind")
|
||
|
coder.encode(members, forKey: "members")
|
||
|
case .usersRemoved(let members):
|
||
|
coder.encode("usersRemoved", forKey: "kind")
|
||
|
coder.encode(members, forKey: "members")
|
||
|
case .userLeft:
|
||
|
coder.encode("userLeft", forKey: "kind")
|
||
4 years ago
|
}
|
||
|
}
|
||
|
|
||
|
// MARK: Proto Conversion
|
||
4 years ago
|
public override class func fromProto(_ proto: SNProtoContent) -> ClosedGroupControlMessage? {
|
||
|
guard let closedGroupControlMessageProto = proto.dataMessage?.closedGroupControlMessage else { return nil }
|
||
4 years ago
|
let kind: Kind
|
||
4 years ago
|
switch closedGroupControlMessageProto.type {
|
||
4 years ago
|
case .new:
|
||
4 years ago
|
guard let publicKey = closedGroupControlMessageProto.publicKey, let name = closedGroupControlMessageProto.name,
|
||
|
let encryptionKeyPairAsProto = closedGroupControlMessageProto.encryptionKeyPair else { return nil }
|
||
4 years ago
|
do {
|
||
4 years ago
|
let encryptionKeyPair = try ECKeyPair(publicKeyData: encryptionKeyPairAsProto.publicKey.removing05PrefixIfNeeded(), privateKeyData: encryptionKeyPairAsProto.privateKey)
|
||
4 years ago
|
kind = .new(publicKey: publicKey, name: name, encryptionKeyPair: encryptionKeyPair,
|
||
4 years ago
|
members: closedGroupControlMessageProto.members, admins: closedGroupControlMessageProto.admins)
|
||
4 years ago
|
} catch {
|
||
|
SNLog("Couldn't parse key pair.")
|
||
|
return nil
|
||
|
}
|
||
|
case .update:
|
||
4 years ago
|
guard let name = closedGroupControlMessageProto.name else { return nil }
|
||
|
kind = .update(name: name, members: closedGroupControlMessageProto.members)
|
||
4 years ago
|
case .encryptionKeyPair:
|
||
4 years ago
|
let wrappers = closedGroupControlMessageProto.wrappers.compactMap { KeyPairWrapper.fromProto($0) }
|
||
4 years ago
|
kind = .encryptionKeyPair(wrappers)
|
||
4 years ago
|
case .nameChange:
|
||
4 years ago
|
guard let name = closedGroupControlMessageProto.name else { return nil }
|
||
4 years ago
|
kind = .nameChange(name: name)
|
||
|
case .usersAdded:
|
||
4 years ago
|
kind = .usersAdded(members: closedGroupControlMessageProto.members)
|
||
4 years ago
|
case .usersRemoved:
|
||
4 years ago
|
kind = .usersRemoved(members: closedGroupControlMessageProto.members)
|
||
4 years ago
|
case .userLeft:
|
||
|
kind = .userLeft
|
||
4 years ago
|
}
|
||
4 years ago
|
return ClosedGroupControlMessage(kind: kind)
|
||
4 years ago
|
}
|
||
|
|
||
|
public override func toProto(using transaction: YapDatabaseReadWriteTransaction) -> SNProtoContent? {
|
||
|
guard let kind = kind else {
|
||
|
SNLog("Couldn't construct closed group update proto from: \(self).")
|
||
|
return nil
|
||
|
}
|
||
|
do {
|
||
4 years ago
|
let closedGroupControlMessage: SNProtoDataMessageClosedGroupControlMessage.SNProtoDataMessageClosedGroupControlMessageBuilder
|
||
4 years ago
|
switch kind {
|
||
|
case .new(let publicKey, let name, let encryptionKeyPair, let members, let admins):
|
||
4 years ago
|
closedGroupControlMessage = SNProtoDataMessageClosedGroupControlMessage.builder(type: .new)
|
||
|
closedGroupControlMessage.setPublicKey(publicKey)
|
||
|
closedGroupControlMessage.setName(name)
|
||
|
let encryptionKeyPairAsProto = SNProtoDataMessageClosedGroupControlMessageKeyPair.builder(publicKey: encryptionKeyPair.publicKey, privateKey: encryptionKeyPair.privateKey)
|
||
4 years ago
|
do {
|
||
4 years ago
|
closedGroupControlMessage.setEncryptionKeyPair(try encryptionKeyPairAsProto.build())
|
||
4 years ago
|
} catch {
|
||
|
SNLog("Couldn't construct closed group update proto from: \(self).")
|
||
|
return nil
|
||
|
}
|
||
4 years ago
|
closedGroupControlMessage.setMembers(members)
|
||
|
closedGroupControlMessage.setAdmins(admins)
|
||
4 years ago
|
case .update(let name, let members):
|
||
4 years ago
|
closedGroupControlMessage = SNProtoDataMessageClosedGroupControlMessage.builder(type: .update)
|
||
|
closedGroupControlMessage.setName(name)
|
||
|
closedGroupControlMessage.setMembers(members)
|
||
4 years ago
|
case .encryptionKeyPair(let wrappers):
|
||
4 years ago
|
closedGroupControlMessage = SNProtoDataMessageClosedGroupControlMessage.builder(type: .encryptionKeyPair)
|
||
|
closedGroupControlMessage.setWrappers(wrappers.compactMap { $0.toProto() })
|
||
4 years ago
|
case .nameChange(let name):
|
||
4 years ago
|
closedGroupControlMessage = SNProtoDataMessageClosedGroupControlMessage.builder(type: .nameChange)
|
||
|
closedGroupControlMessage.setName(name)
|
||
4 years ago
|
case .usersAdded(let members):
|
||
4 years ago
|
closedGroupControlMessage = SNProtoDataMessageClosedGroupControlMessage.builder(type: .usersAdded)
|
||
|
closedGroupControlMessage.setMembers(members)
|
||
4 years ago
|
case .usersRemoved(let members):
|
||
4 years ago
|
closedGroupControlMessage = SNProtoDataMessageClosedGroupControlMessage.builder(type: .usersRemoved)
|
||
|
closedGroupControlMessage.setMembers(members)
|
||
4 years ago
|
case .userLeft:
|
||
4 years ago
|
closedGroupControlMessage = SNProtoDataMessageClosedGroupControlMessage.builder(type: .userLeft)
|
||
4 years ago
|
}
|
||
|
let contentProto = SNProtoContent.builder()
|
||
|
let dataMessageProto = SNProtoDataMessage.builder()
|
||
4 years ago
|
dataMessageProto.setClosedGroupControlMessage(try closedGroupControlMessage.build())
|
||
4 years ago
|
// Group context
|
||
|
try setGroupContextIfNeeded(on: dataMessageProto, using: transaction)
|
||
|
// Expiration timer
|
||
|
// TODO: We * want * expiration timer updates to be explicit. But currently Android will disable the expiration timer for a conversation
|
||
|
// if it receives a message without the current expiration timer value attached to it...
|
||
|
var expiration: UInt32 = 0
|
||
|
if let disappearingMessagesConfiguration = OWSDisappearingMessagesConfiguration.fetch(uniqueId: threadID!, transaction: transaction) {
|
||
|
expiration = disappearingMessagesConfiguration.isEnabled ? disappearingMessagesConfiguration.durationSeconds : 0
|
||
|
}
|
||
|
dataMessageProto.setExpireTimer(expiration)
|
||
|
contentProto.setDataMessage(try dataMessageProto.build())
|
||
|
return try contentProto.build()
|
||
|
} catch {
|
||
|
SNLog("Couldn't construct closed group update proto from: \(self).")
|
||
|
return nil
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// MARK: Description
|
||
|
public override var description: String {
|
||
|
"""
|
||
4 years ago
|
ClosedGroupControlMessage(
|
||
4 years ago
|
kind: \(kind?.description ?? "null")
|
||
|
)
|
||
|
"""
|
||
|
}
|
||
|
}
|