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.
		
		
		
		
		
			
		
			
				
	
	
		
			148 lines
		
	
	
		
			7.5 KiB
		
	
	
	
		
			Swift
		
	
			
		
		
	
	
			148 lines
		
	
	
		
			7.5 KiB
		
	
	
	
		
			Swift
		
	
| import SessionProtocolKit
 | |
| 
 | |
| extension Storage {
 | |
|     
 | |
|     // MARK: - V2
 | |
|     
 | |
|     private static func getClosedGroupEncryptionKeyPairCollection(for groupPublicKey: String) -> String {
 | |
|         return "SNClosedGroupEncryptionKeyPairCollection-\(groupPublicKey)"
 | |
|     }
 | |
| 
 | |
|     private static let closedGroupPublicKeyCollection = "SNClosedGroupPublicKeyCollection"
 | |
|     
 | |
|     private static let closedGroupFormationTimestampCollection = "SNClosedGroupFormationTimestampCollection"
 | |
| 
 | |
|     public func getClosedGroupEncryptionKeyPairs(for groupPublicKey: String) -> [ECKeyPair] {
 | |
|         let collection = Storage.getClosedGroupEncryptionKeyPairCollection(for: groupPublicKey)
 | |
|         var timestampsAndKeyPairs: [(timestamp: Double, keyPair: ECKeyPair)] = []
 | |
|         Storage.read { transaction in
 | |
|             transaction.enumerateKeysAndObjects(inCollection: collection) { key, object, _ in
 | |
|                 guard let timestamp = Double(key), let keyPair = object as? ECKeyPair else { return }
 | |
|                 timestampsAndKeyPairs.append((timestamp, keyPair))
 | |
|             }
 | |
|         }
 | |
|         return timestampsAndKeyPairs.sorted { $0.timestamp < $1.timestamp }.map { $0.keyPair }
 | |
|     }
 | |
| 
 | |
|     public func getLatestClosedGroupEncryptionKeyPair(for groupPublicKey: String) -> ECKeyPair? {
 | |
|         return getClosedGroupEncryptionKeyPairs(for: groupPublicKey).last
 | |
|     }
 | |
| 
 | |
|     public func addClosedGroupEncryptionKeyPair(_ keyPair: ECKeyPair, for groupPublicKey: String, using transaction: Any) {
 | |
|         let collection = Storage.getClosedGroupEncryptionKeyPairCollection(for: groupPublicKey)
 | |
|         let timestamp = String(Date().timeIntervalSince1970)
 | |
|         (transaction as! YapDatabaseReadWriteTransaction).setObject(keyPair, forKey: timestamp, inCollection: collection)
 | |
|     }
 | |
| 
 | |
|     public func removeAllClosedGroupEncryptionKeyPairs(for groupPublicKey: String, using transaction: Any) {
 | |
|         let collection = Storage.getClosedGroupEncryptionKeyPairCollection(for: groupPublicKey)
 | |
|         (transaction as! YapDatabaseReadWriteTransaction).removeAllObjects(inCollection: collection)
 | |
|     }
 | |
| 
 | |
|     public func addClosedGroupPublicKey(_ groupPublicKey: String, using transaction: Any) {
 | |
|         (transaction as! YapDatabaseReadWriteTransaction).setObject(groupPublicKey, forKey: groupPublicKey, inCollection: Storage.closedGroupPublicKeyCollection)
 | |
|     }
 | |
|     
 | |
|     public func removeClosedGroupPublicKey(_ groupPublicKey: String, using transaction: Any) {
 | |
|         (transaction as! YapDatabaseReadWriteTransaction).removeObject(forKey: groupPublicKey, inCollection: Storage.closedGroupPublicKeyCollection)
 | |
|     }
 | |
|     
 | |
|     public func getClosedGroupFormationTimestamp(for groupPublicKey: String) -> UInt64? {
 | |
|         var result: UInt64?
 | |
|         Storage.read { transaction in
 | |
|             result = transaction.object(forKey: groupPublicKey, inCollection: Storage.closedGroupFormationTimestampCollection) as? UInt64
 | |
|         }
 | |
|         return result
 | |
|     }
 | |
|     
 | |
|     public func setClosedGroupFormationTimestamp(to timestamp: UInt64, for groupPublicKey: String, using transaction: Any) {
 | |
|         (transaction as! YapDatabaseReadWriteTransaction).setObject(timestamp, forKey: groupPublicKey, inCollection: Storage.closedGroupFormationTimestampCollection)
 | |
|     }
 | |
|     
 | |
|     
 | |
|     
 | |
|     // MARK: - Ratchets
 | |
|     
 | |
|     private static func getClosedGroupRatchetCollection(_ collection: ClosedGroupRatchetCollectionType, for groupPublicKey: String) -> String {
 | |
|         switch collection {
 | |
|         case .old: return "LokiOldClosedGroupRatchetCollection.\(groupPublicKey)"
 | |
|         case .current: return "LokiClosedGroupRatchetCollection.\(groupPublicKey)"
 | |
|         }
 | |
|     }
 | |
|     
 | |
|     public func getClosedGroupRatchet(for groupPublicKey: String, senderPublicKey: String, from collection: ClosedGroupRatchetCollectionType = .current) -> ClosedGroupRatchet? {
 | |
|         let collection = Storage.getClosedGroupRatchetCollection(collection, for: groupPublicKey)
 | |
|         var result: ClosedGroupRatchet?
 | |
|         Storage.read { transaction in
 | |
|             result = transaction.object(forKey: senderPublicKey, inCollection: collection) as? ClosedGroupRatchet
 | |
|         }
 | |
|         return result
 | |
|     }
 | |
| 
 | |
|     public func setClosedGroupRatchet(for groupPublicKey: String, senderPublicKey: String, ratchet: ClosedGroupRatchet, in collection: ClosedGroupRatchetCollectionType = .current, using transaction: Any) {
 | |
|         let collection = Storage.getClosedGroupRatchetCollection(collection, for: groupPublicKey)
 | |
|         (transaction as! YapDatabaseReadWriteTransaction).setObject(ratchet, forKey: senderPublicKey, inCollection: collection)
 | |
|     }
 | |
|     
 | |
|     public func getAllClosedGroupRatchets(for groupPublicKey: String, from collection: ClosedGroupRatchetCollectionType = .current) -> [(senderPublicKey: String, ratchet: ClosedGroupRatchet)] {
 | |
|         let collection = Storage.getClosedGroupRatchetCollection(collection, for: groupPublicKey)
 | |
|         var result: [(senderPublicKey: String, ratchet: ClosedGroupRatchet)] = []
 | |
|         Storage.read { transaction in
 | |
|             transaction.enumerateRows(inCollection: collection) { key, object, _, _ in
 | |
|                 guard let ratchet = object as? ClosedGroupRatchet else { return }
 | |
|                 let senderPublicKey = key
 | |
|                 result.append((senderPublicKey: senderPublicKey, ratchet: ratchet))
 | |
|             }
 | |
|         }
 | |
|         return result
 | |
|     }
 | |
| 
 | |
|     public func removeAllClosedGroupRatchets(for groupPublicKey: String, from collection: ClosedGroupRatchetCollectionType = .current, using transaction: Any) {
 | |
|         let collection = Storage.getClosedGroupRatchetCollection(collection, for: groupPublicKey)
 | |
|         (transaction as! YapDatabaseReadWriteTransaction).removeAllObjects(inCollection: collection)
 | |
|     }
 | |
|     
 | |
|     // MARK: - Private Keys
 | |
|     
 | |
|     private static let closedGroupPrivateKeyCollection = "LokiClosedGroupPrivateKeyCollection"
 | |
| 
 | |
|     public func getClosedGroupPrivateKey(for publicKey: String) -> String? {
 | |
|         var result: String?
 | |
|         Storage.read { transaction in
 | |
|             result = transaction.object(forKey: publicKey, inCollection: Storage.closedGroupPrivateKeyCollection) as? String
 | |
|         }
 | |
|         return result
 | |
|     }
 | |
| 
 | |
|     public func setClosedGroupPrivateKey(_ privateKey: String, for publicKey: String, using transaction: Any) {
 | |
|         (transaction as! YapDatabaseReadWriteTransaction).setObject(privateKey, forKey: publicKey, inCollection: Storage.closedGroupPrivateKeyCollection)
 | |
|     }
 | |
| 
 | |
|     public func removeClosedGroupPrivateKey(for publicKey: String, using transaction: Any) {
 | |
|         (transaction as! YapDatabaseReadWriteTransaction).removeObject(forKey: publicKey, inCollection: Storage.closedGroupPrivateKeyCollection)
 | |
|     }
 | |
| 
 | |
|     
 | |
|     
 | |
|     // MARK: - Convenience
 | |
|     
 | |
|     public func getAllClosedGroupSenderKeys(for groupPublicKey: String, from collection: ClosedGroupRatchetCollectionType = .current) -> Set<ClosedGroupSenderKey> {
 | |
|         return Set(getAllClosedGroupRatchets(for: groupPublicKey, from: collection).map { senderPublicKey, ratchet in
 | |
|             ClosedGroupSenderKey(chainKey: Data(hex: ratchet.chainKey), keyIndex: ratchet.keyIndex, publicKey: Data(hex: senderPublicKey))
 | |
|         })
 | |
|     }
 | |
|     
 | |
|     public func getUserClosedGroupPublicKeys() -> Set<String> {
 | |
|         var result: Set<String> = []
 | |
|         Storage.read { transaction in
 | |
|             result = result.union(Set(transaction.allKeys(inCollection: Storage.closedGroupPublicKeyCollection)))
 | |
|             result = result.union(Set(transaction.allKeys(inCollection: Storage.closedGroupPrivateKeyCollection)))
 | |
|         }
 | |
|         return result
 | |
|     }
 | |
| 
 | |
|     public func isClosedGroup(_ publicKey: String) -> Bool {
 | |
|         getUserClosedGroupPublicKeys().contains(publicKey)
 | |
|     }
 | |
| }
 |