|  |  |  | // Copyright © 2023 Rangeproof Pty Ltd. All rights reserved. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import Foundation | 
					
						
							|  |  |  | import GRDB | 
					
						
							|  |  |  | import SessionUtilitiesKit | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /// This migration adds a table to track pending read receipts (it's possible to receive a read receipt message before getting the original | 
					
						
							|  |  |  | /// message due to how one-to-one conversations work, by storing pending read receipts we should be able to prevent this case) | 
					
						
							|  |  |  | enum _011_AddPendingReadReceipts: Migration { | 
					
						
							|  |  |  |     static let target: TargetMigrations.Identifier = .messagingKit | 
					
						
							|  |  |  |     static let identifier: String = "AddPendingReadReceipts" // stringlint:disable | 
					
						
							|  |  |  |     static let needsConfigSync: Bool = false | 
					
						
							|  |  |  |     static let minExpectedRunDuration: TimeInterval = 0.01 | 
					
						
							|  |  |  |     static let fetchedTables: [(TableRecord & FetchableRecord).Type] = [] | 
					
						
							|  |  |  |     static let createdOrAlteredTables: [(TableRecord & FetchableRecord).Type] = [PendingReadReceipt.self] | 
					
						
							|  |  |  |     static let droppedTables: [(TableRecord & FetchableRecord).Type] = [] | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     static func migrate(_ db: Database) throws { | 
					
						
							|  |  |  |         try db.create(table: PendingReadReceipt.self) { t in | 
					
						
							|  |  |  |             t.column(.threadId, .text) | 
					
						
							|  |  |  |                 .notNull() | 
					
						
							|  |  |  |                 .indexed()                                            // Quicker querying | 
					
						
							|  |  |  |                 .references(SessionThread.self, onDelete: .cascade)   // Delete if Thread deleted | 
					
						
							|  |  |  |             t.column(.interactionTimestampMs, .integer) | 
					
						
							|  |  |  |                 .notNull() | 
					
						
							|  |  |  |                 .indexed()                                            // Quicker querying | 
					
						
							|  |  |  |             t.column(.readTimestampMs, .integer) | 
					
						
							|  |  |  |                 .notNull() | 
					
						
							|  |  |  |             t.column(.serverExpirationTimestamp, .double) | 
					
						
							|  |  |  |                 .notNull() | 
					
						
							|  |  |  |              | 
					
						
							|  |  |  |             t.primaryKey([.threadId, .interactionTimestampMs]) | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |          | 
					
						
							|  |  |  |         Storage.update(progress: 1, for: self, in: target) // In case this is the last migration | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } |