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.
175 lines
8.1 KiB
Swift
175 lines
8.1 KiB
Swift
// Copyright © 2022 Rangeproof Pty Ltd. All rights reserved.
|
|
|
|
import Foundation
|
|
import GRDB
|
|
import SessionUtilitiesKit
|
|
|
|
public struct SnodeReceivedMessageInfo: Codable, FetchableRecord, MutablePersistableRecord, TableRecord, ColumnExpressible {
|
|
public static var databaseTableName: String { "snodeReceivedMessageInfo" }
|
|
|
|
public typealias Columns = CodingKeys
|
|
public enum CodingKeys: String, CodingKey, ColumnExpression {
|
|
case id
|
|
case key
|
|
case hash
|
|
case expirationDateMs
|
|
case wasDeletedOrInvalid
|
|
}
|
|
|
|
/// The `id` value is auto incremented by the database, if the `Job` hasn't been inserted into
|
|
/// the database yet this value will be `nil`
|
|
public var id: Int64? = nil
|
|
|
|
/// The key this message hash is associated to
|
|
///
|
|
/// This will be a combination of {address}.{port}.{publicKey} for new rows and just the {publicKey} for legacy rows
|
|
public let key: String
|
|
|
|
/// The is the hash for the received message
|
|
public let hash: String
|
|
|
|
/// This is the timestamp (in milliseconds since epoch) when the message hash should expire
|
|
///
|
|
/// **Note:** If no value exists this will default to 15 days from now (since the service node caches messages for
|
|
/// 14 days)
|
|
public let expirationDateMs: Int64
|
|
|
|
/// This flag indicates whether the interaction associated with this message hash was deleted or whether this message
|
|
/// hash is potentially invalid (if a poll results in 100% of the `SnodeReceivedMessageInfo` entries being seen as
|
|
/// duplicates then we assume that the `lastHash` value provided when retrieving messages was invalid and mark
|
|
/// it as such)
|
|
///
|
|
/// **Note:** When retrieving the `lastNotExpired` we will ignore any entries where this flag is true
|
|
public var wasDeletedOrInvalid: Bool?
|
|
|
|
// MARK: - Custom Database Interaction
|
|
|
|
public mutating func didInsert(_ inserted: InsertionSuccess) {
|
|
self.id = inserted.rowID
|
|
}
|
|
}
|
|
|
|
// MARK: - Convenience
|
|
|
|
public extension SnodeReceivedMessageInfo {
|
|
private static func key(for snode: Snode, publicKey: String, namespace: Int) -> String {
|
|
guard namespace != SnodeAPI.defaultNamespace else {
|
|
return "\(snode.address):\(snode.port).\(publicKey)"
|
|
}
|
|
|
|
return "\(snode.address):\(snode.port).\(publicKey).\(namespace)"
|
|
}
|
|
|
|
init(
|
|
snode: Snode,
|
|
publicKey: String,
|
|
namespace: Int,
|
|
hash: String,
|
|
expirationDateMs: Int64?
|
|
) {
|
|
self.key = SnodeReceivedMessageInfo.key(for: snode, publicKey: publicKey, namespace: namespace)
|
|
self.hash = hash
|
|
self.expirationDateMs = (expirationDateMs ?? 0)
|
|
}
|
|
}
|
|
|
|
// MARK: - GRDB Interactions
|
|
|
|
public extension SnodeReceivedMessageInfo {
|
|
static func pruneExpiredMessageHashInfo(for snode: Snode, namespace: Int, associatedWith publicKey: String) {
|
|
// Delete any expired SnodeReceivedMessageInfo values associated to a specific node (even though
|
|
// this runs very quickly we fetch the rowIds we want to delete from a 'read' call to avoid
|
|
// blocking the write queue since this method is called very frequently)
|
|
let rowIds: [Int64] = Storage.shared
|
|
.read { db in
|
|
// Only prune the hashes if new hashes exist for this Snode (if they don't then we don't want
|
|
// to clear out the legacy hashes)
|
|
let hasNonLegacyHash: Bool = try SnodeReceivedMessageInfo
|
|
.filter(SnodeReceivedMessageInfo.Columns.key == key(for: snode, publicKey: publicKey, namespace: namespace))
|
|
.isNotEmpty(db)
|
|
|
|
guard hasNonLegacyHash else { return [] }
|
|
|
|
return try SnodeReceivedMessageInfo
|
|
.select(Column.rowID)
|
|
.filter(SnodeReceivedMessageInfo.Columns.key == key(for: snode, publicKey: publicKey, namespace: namespace))
|
|
.filter(SnodeReceivedMessageInfo.Columns.expirationDateMs <= (Date().timeIntervalSince1970 * 1000))
|
|
.asRequest(of: Int64.self)
|
|
.fetchAll(db)
|
|
}
|
|
.defaulting(to: [])
|
|
|
|
// If there are no rowIds to delete then do nothing
|
|
guard !rowIds.isEmpty else { return }
|
|
|
|
Storage.shared.write { db in
|
|
try SnodeReceivedMessageInfo
|
|
.filter(rowIds.contains(Column.rowID))
|
|
.deleteAll(db)
|
|
}
|
|
}
|
|
|
|
/// This method fetches the last non-expired hash from the database for message retrieval
|
|
///
|
|
/// **Note:** This method uses a `write` instead of a read because there is a single write queue for the database and it's very common for
|
|
/// this method to be called after the hash value has been updated but before the various `read` threads have been updated, resulting in a
|
|
/// pointless fetch for data the app has already received
|
|
static func fetchLastNotExpired(for snode: Snode, namespace: Int, associatedWith publicKey: String) -> SnodeReceivedMessageInfo? {
|
|
return Storage.shared.read { db in
|
|
let nonLegacyHash: SnodeReceivedMessageInfo? = try SnodeReceivedMessageInfo
|
|
.filter(
|
|
SnodeReceivedMessageInfo.Columns.wasDeletedOrInvalid == nil ||
|
|
SnodeReceivedMessageInfo.Columns.wasDeletedOrInvalid == false
|
|
)
|
|
.filter(SnodeReceivedMessageInfo.Columns.key == key(for: snode, publicKey: publicKey, namespace: namespace))
|
|
.filter(SnodeReceivedMessageInfo.Columns.expirationDateMs > (Date().timeIntervalSince1970 * 1000))
|
|
.order(SnodeReceivedMessageInfo.Columns.id.desc)
|
|
.fetchOne(db)
|
|
|
|
// If we have a non-legacy hash then return it immediately (legacy hashes had a different
|
|
// 'key' structure)
|
|
if nonLegacyHash != nil { return nonLegacyHash }
|
|
|
|
return try SnodeReceivedMessageInfo
|
|
.filter(
|
|
SnodeReceivedMessageInfo.Columns.wasDeletedOrInvalid == nil ||
|
|
SnodeReceivedMessageInfo.Columns.wasDeletedOrInvalid == false
|
|
)
|
|
.filter(SnodeReceivedMessageInfo.Columns.key == publicKey)
|
|
.order(SnodeReceivedMessageInfo.Columns.id.desc)
|
|
.fetchOne(db)
|
|
}
|
|
}
|
|
|
|
/// There are some cases where the latest message can be removed from a swarm, if we then try to poll for that message the swarm
|
|
/// will see it as invalid and start returning messages from the beginning which can result in a lot of wasted, duplicate downloads
|
|
///
|
|
/// This method should be called when deleting a message, handling an UnsendRequest or when receiving a poll response which contains
|
|
/// solely duplicate messages (for the specific service node - if even one message in a response is new for that service node then this shouldn't
|
|
/// be called if if the message has already been received and processed by a separate service node)
|
|
static func handlePotentialDeletedOrInvalidHash(
|
|
_ db: Database,
|
|
potentiallyInvalidHashes: [String],
|
|
otherKnownValidHashes: [String] = []
|
|
) throws {
|
|
_ = try SnodeReceivedMessageInfo
|
|
.filter(potentiallyInvalidHashes.contains(SnodeReceivedMessageInfo.Columns.hash))
|
|
.updateAll(
|
|
db,
|
|
SnodeReceivedMessageInfo.Columns.wasDeletedOrInvalid.set(to: true)
|
|
)
|
|
|
|
// If we have any server hashes which we know are valid (eg. we fetched the oldest messages) then
|
|
// mark them all as valid to prevent the case where we just slowly work backwards from the latest
|
|
// message, polling for one earlier each time
|
|
guard !otherKnownValidHashes.isEmpty else { return }
|
|
|
|
_ = try SnodeReceivedMessageInfo
|
|
.filter(otherKnownValidHashes.contains(SnodeReceivedMessageInfo.Columns.hash))
|
|
.updateAll(
|
|
db,
|
|
SnodeReceivedMessageInfo.Columns.wasDeletedOrInvalid.set(to: false)
|
|
)
|
|
}
|
|
}
|