Swiftgram/submodules/Postbox/Sources/PendingMessageActionsMetadataTable.swift
2019-11-01 17:11:12 +04:00

114 lines
4.3 KiB
Swift

import Foundation
enum PendingMessageActionsMetadataCountKey: Hashable {
case peerNamespace(PeerId, MessageId.Namespace)
case peerNamespaceAction(PeerId, MessageId.Namespace, PendingMessageActionType)
static func ==(lhs: PendingMessageActionsMetadataCountKey, rhs: PendingMessageActionsMetadataCountKey) -> Bool {
switch lhs {
case let .peerNamespace(peerId, namespace):
if case .peerNamespace(peerId, namespace) = rhs {
return true
} else {
return false
}
case let .peerNamespaceAction(peerId, namespace, actionType):
if case .peerNamespaceAction(peerId, namespace, actionType) = rhs {
return true
} else {
return false
}
}
}
var hashValue: Int {
switch self {
case let .peerNamespace(peerId, namespace):
return peerId.hashValue ^ namespace.hashValue
case let .peerNamespaceAction(peerId, namespace, actionType):
return peerId.hashValue ^ namespace.hashValue ^ actionType.hashValue
}
}
}
enum PendingMessageActionsMetadataKey {
case count(PendingMessageActionsMetadataCountKey)
}
final class PendingMessageActionsMetadataTable: Table {
static func tableSpec(_ id: Int32) -> ValueBoxTable {
return ValueBoxTable(id: id, keyType: .binary, compactValuesOnCreation: true)
}
private var cachedCounts: [PendingMessageActionsMetadataCountKey: Int32] = [:]
private var updatedCounts = Set<PendingMessageActionsMetadataCountKey>()
private func key(_ key: PendingMessageActionsMetadataKey) -> ValueBoxKey {
switch key {
case let .count(countKey):
switch countKey {
case let .peerNamespace(peerId, namespace):
let result = ValueBoxKey(length: 1 + 8 + 4)
result.setUInt8(0, value: 0)
result.setInt64(1, value: peerId.toInt64())
result.setInt32(1 + 8, value: namespace)
return result
case let .peerNamespaceAction(peerId, namespace, actionType):
let result = ValueBoxKey(length: 1 + 8 + 4 + 4)
result.setUInt8(0, value: 1)
result.setInt64(1, value: peerId.toInt64())
result.setInt32(1 + 8, value: namespace)
result.setUInt32(1 + 8 + 4, value: actionType.rawValue)
return result
}
}
}
func getCount(_ key: PendingMessageActionsMetadataCountKey) -> Int32 {
if let cached = self.cachedCounts[key] {
return cached
} else if let value = self.valueBox.get(self.table, key: self.key(.count(key))) {
var count: Int32 = 0
value.read(&count, offset: 0, length: 4)
self.cachedCounts[key] = count
return count
} else {
self.cachedCounts[key] = 0
return 0
}
}
func setCount(_ key: PendingMessageActionsMetadataCountKey, value: Int32) {
if self.cachedCounts[key] != value {
self.cachedCounts[key] = value
self.updatedCounts.insert(key)
}
}
func addCount(_ key: PendingMessageActionsMetadataCountKey, value: Int32) -> Int32 {
let updatedCount = self.getCount(key) + value
self.setCount(key, value: updatedCount)
return updatedCount
}
override func clearMemoryCache() {
self.cachedCounts.removeAll()
assert(self.updatedCounts.isEmpty)
}
override func beforeCommit() {
if !self.updatedCounts.isEmpty {
for key in self.updatedCounts {
if let count = self.cachedCounts[key] {
var countValue: Int32 = count
self.valueBox.set(self.table, key: self.key(.count(key)), value: MemoryBuffer(memory: &countValue, capacity: 4, length: 4, freeWhenDone: false))
} else {
assertionFailure()
}
}
self.updatedCounts.removeAll()
}
}
}