|  |  |  | // | 
					
						
							|  |  |  | //  Copyright (c) 2018 Open Whisper Systems. All rights reserved. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @objc | 
					
						
							|  |  |  | public class AnyLRUCache: NSObject { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     private let backingCache: LRUCache<NSObject, NSObject> | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @objc | 
					
						
							|  |  |  |     public init(maxSize: Int) { | 
					
						
							|  |  |  |         backingCache = LRUCache(maxSize: maxSize) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @objc | 
					
						
							|  |  |  |     public func get(key: NSObject) -> NSObject? { | 
					
						
							|  |  |  |         return self.backingCache.get(key: key) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @objc | 
					
						
							|  |  |  |     public func set(key: NSObject, value: NSObject) { | 
					
						
							|  |  |  |         self.backingCache.set(key: key, value: value) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @objc | 
					
						
							|  |  |  |     public func clear() { | 
					
						
							|  |  |  |         self.backingCache.clear() | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // A simple LRU cache bounded by the number of entries. | 
					
						
							|  |  |  | public class LRUCache<KeyType: Hashable & Equatable, ValueType> { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     private var cacheMap: [KeyType: ValueType] = [:] | 
					
						
							|  |  |  |     private var cacheOrder: [KeyType] = [] | 
					
						
							|  |  |  |     private let maxSize: Int | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @objc | 
					
						
							|  |  |  |     public init(maxSize: Int) { | 
					
						
							|  |  |  |         self.maxSize = maxSize | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         NotificationCenter.default.addObserver(self, | 
					
						
							|  |  |  |                                                selector: #selector(didReceiveMemoryWarning), | 
					
						
							|  |  |  |                                                name: NSNotification.Name.UIApplicationDidReceiveMemoryWarning, | 
					
						
							|  |  |  |                                                object: nil) | 
					
						
							|  |  |  |         NotificationCenter.default.addObserver(self, | 
					
						
							|  |  |  |                                                selector: #selector(didEnterBackground), | 
					
						
							|  |  |  |                                                name: NSNotification.Name.OWSApplicationDidEnterBackground, | 
					
						
							|  |  |  |                                                object: nil) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     deinit { | 
					
						
							|  |  |  |         NotificationCenter.default.removeObserver(self) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @objc func didEnterBackground() { | 
					
						
							|  |  |  |         AssertIsOnMainThread() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         clear() | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @objc func didReceiveMemoryWarning() { | 
					
						
							|  |  |  |         AssertIsOnMainThread() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         clear() | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     private func updateCacheOrder(key: KeyType) { | 
					
						
							|  |  |  |         cacheOrder = cacheOrder.filter { $0 != key } | 
					
						
							|  |  |  |         cacheOrder.append(key) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     public func get(key: KeyType) -> ValueType? { | 
					
						
							|  |  |  |         guard let value = cacheMap[key] else { | 
					
						
							|  |  |  |             // Miss | 
					
						
							|  |  |  |             return nil | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Hit | 
					
						
							|  |  |  |         updateCacheOrder(key: key) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return value | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     public func set(key: KeyType, value: ValueType) { | 
					
						
							|  |  |  |         cacheMap[key] = value | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         updateCacheOrder(key: key) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         while cacheOrder.count > maxSize { | 
					
						
							|  |  |  |             guard let staleKey = cacheOrder.first else { | 
					
						
							|  |  |  |                 owsFail("Cache ordering unexpectedly empty") | 
					
						
							|  |  |  |                 return | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             cacheOrder.removeFirst() | 
					
						
							|  |  |  |             cacheMap.removeValue(forKey: staleKey) | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @objc | 
					
						
							|  |  |  |     public func clear() { | 
					
						
							|  |  |  |         cacheMap.removeAll() | 
					
						
							|  |  |  |         cacheOrder.removeAll() | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } |