// Copyright © 2022 Rangeproof Pty Ltd. All rights reserved. import Foundation import GRDB import SessionUtilitiesKit enum _003_YDBToGRDBMigration: Migration { static let identifier: String = "YDBToGRDBMigration" static let minExpectedRunDuration: TimeInterval = 0.2 static let needsConfigSync: Bool = false static func migrate(_ db: Database) throws { // MARK: - OnionRequestPath, Snode Pool & Swarm // Note: Want to exclude the Snode's we already added from the 'onionRequestPathResult' var snodeResult: Set = [] var snodeSetResult: [String: Set] = [:] var lastSnodePoolRefreshDate: Date? = nil // Map the Legacy types for the NSKeyedUnarchiver NSKeyedUnarchiver.setClass( SSKLegacy.Snode.self, forClassName: "SessionSnodeKit.Snode" ) Storage.read { transaction in // Process the lastSnodePoolRefreshDate lastSnodePoolRefreshDate = transaction.object( forKey: SSKLegacy.lastSnodePoolRefreshDateKey, inCollection: SSKLegacy.lastSnodePoolRefreshDateCollection ) as? Date // Process the OnionRequestPaths if let path0Snode0 = transaction.object(forKey: "0-0", inCollection: SSKLegacy.onionRequestPathCollection) as? SSKLegacy.Snode, let path0Snode1 = transaction.object(forKey: "0-1", inCollection: SSKLegacy.onionRequestPathCollection) as? SSKLegacy.Snode, let path0Snode2 = transaction.object(forKey: "0-2", inCollection: SSKLegacy.onionRequestPathCollection) as? SSKLegacy.Snode { snodeResult.insert(path0Snode0) snodeResult.insert(path0Snode1) snodeResult.insert(path0Snode2) snodeSetResult["\(SnodeSet.onionRequestPathPrefix)0"] = [ path0Snode0, path0Snode1, path0Snode2 ] if let path1Snode0 = transaction.object(forKey: "1-0", inCollection: SSKLegacy.onionRequestPathCollection) as? SSKLegacy.Snode, let path1Snode1 = transaction.object(forKey: "1-1", inCollection: SSKLegacy.onionRequestPathCollection) as? SSKLegacy.Snode, let path1Snode2 = transaction.object(forKey: "1-2", inCollection: SSKLegacy.onionRequestPathCollection) as? SSKLegacy.Snode { snodeResult.insert(path1Snode0) snodeResult.insert(path1Snode1) snodeResult.insert(path1Snode2) snodeSetResult["\(SnodeSet.onionRequestPathPrefix)1"] = [ path1Snode0, path1Snode1, path1Snode2 ] } } // Process the SnodePool transaction.enumerateKeysAndObjects(inCollection: SSKLegacy.snodePoolCollection) { _, object, _ in guard let snode = object as? SSKLegacy.Snode else { return } snodeResult.insert(snode) } // Process the Swarms var swarmCollections: Set = [] transaction.enumerateCollections { collectionName, _ in if collectionName.starts(with: SSKLegacy.swarmCollectionPrefix) { swarmCollections.insert(collectionName.substring(from: SSKLegacy.swarmCollectionPrefix.count)) } } for swarmCollection in swarmCollections { let collection: String = "\(SSKLegacy.swarmCollectionPrefix)\(swarmCollection)" transaction.enumerateKeysAndObjects(inCollection: collection) { _, object, _ in guard let snode = object as? SSKLegacy.Snode else { return } snodeResult.insert(snode) snodeSetResult[swarmCollection] = (snodeSetResult[swarmCollection] ?? Set()).inserting(snode) } } } // Insert the data into GRDB try autoreleasepool { db[.lastSnodePoolRefreshDate] = lastSnodePoolRefreshDate try snodeResult.forEach { legacySnode in try Snode( address: legacySnode.address, port: legacySnode.port, ed25519PublicKey: legacySnode.publicKeySet.ed25519Key, x25519PublicKey: legacySnode.publicKeySet.x25519Key ).insert(db) } try snodeSetResult.forEach { key, legacySnodeSet in try legacySnodeSet.enumerated().forEach { nodeIndex, legacySnode in // Note: In this case the 'nodeIndex' is irrelivant try SnodeSet( key: key, nodeIndex: nodeIndex, address: legacySnode.address, port: legacySnode.port ).insert(db) } } } // MARK: - Received Messages & Last Message Hash var lastMessageResults: [String: (hash: String, json: JSON)] = [:] var receivedMessageResults: [String: Set] = [:] // TODO: Move into the top read block??? Storage.read { transaction in // Extract the received message hashes transaction.enumerateKeysAndObjects(inCollection: SSKLegacy.receivedMessagesCollection) { key, object, _ in guard let hashSet = object as? Set else { return } receivedMessageResults[key] = hashSet } // Retrieve the last message info transaction.enumerateKeysAndObjects(inCollection: SSKLegacy.lastMessageHashCollection) { key, object, _ in guard let lastMessageJson = object as? JSON else { return } guard let lastMessageHash: String = lastMessageJson["hash"] as? String else { return } // Note: We remove the value from 'receivedMessageResults' as we want to try and use // it's actual 'expirationDate' value lastMessageResults[key] = (lastMessageHash, lastMessageJson) receivedMessageResults[key] = receivedMessageResults[key]?.removing(lastMessageHash) } } try autoreleasepool { try receivedMessageResults.forEach { key, hashes in try hashes.forEach { hash in _ = try SnodeReceivedMessageInfo( key: key, hash: hash, expirationDateMs: SnodeReceivedMessage.defaultExpirationSeconds ).inserted(db) } } try lastMessageResults.forEach { key, data in let expirationDateMs: Int64 = ((data.json["expirationDate"] as? Int64) ?? 0) _ = try SnodeReceivedMessageInfo( key: key, hash: data.hash, expirationDateMs: (expirationDateMs > 0 ? expirationDateMs : SnodeReceivedMessage.defaultExpirationSeconds ) ).inserted(db) } } } }