| 
									
										
										
										
											2024-05-29 01:56:00 +10:00
										 |  |  | // Copyright 2024 Signal Messenger, LLC
 | 
					
						
							|  |  |  | // SPDX-License-Identifier: AGPL-3.0-only
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import { assert } from 'chai'; | 
					
						
							|  |  |  | import { v4 as generateGuid } from 'uuid'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import { | 
					
						
							| 
									
										
										
										
											2024-12-04 14:03:29 -08:00
										 |  |  |   dequeueOldestSyncTasks, | 
					
						
							| 
									
										
										
										
											2024-07-22 11:16:33 -07:00
										 |  |  |   removeSyncTaskById, | 
					
						
							|  |  |  |   saveSyncTasks, | 
					
						
							| 
									
										
										
										
											2024-05-29 01:56:00 +10:00
										 |  |  | } from '../../sql/Server'; | 
					
						
							| 
									
										
										
										
											2025-01-16 13:34:35 -08:00
										 |  |  | import type { WritableDB, ReadableDB, MessageType } from '../../sql/Interface'; | 
					
						
							|  |  |  | import { sql, jsonToObject } from '../../sql/util'; | 
					
						
							| 
									
										
										
										
											2025-03-12 14:45:54 -07:00
										 |  |  | import { insertData, updateToVersion, createDB, explain } from './helpers'; | 
					
						
							| 
									
										
										
										
											2024-05-29 01:56:00 +10:00
										 |  |  | import { MAX_SYNC_TASK_ATTEMPTS } from '../../util/syncTasks.types'; | 
					
						
							|  |  |  | import { WEEK } from '../../util/durations'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import type { MessageAttributesType } from '../../model-types'; | 
					
						
							|  |  |  | import type { SyncTaskType } from '../../util/syncTasks'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* eslint-disable camelcase */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-16 13:34:35 -08:00
										 |  |  | // Snapshot before: 1270
 | 
					
						
							|  |  |  | export function getMostRecentAddressableMessages( | 
					
						
							|  |  |  |   db: ReadableDB, | 
					
						
							|  |  |  |   conversationId: string, | 
					
						
							|  |  |  |   limit = 5 | 
					
						
							|  |  |  | ): Array<MessageType> { | 
					
						
							|  |  |  |   const [query, parameters] = sql`
 | 
					
						
							|  |  |  |     SELECT json FROM messages | 
					
						
							|  |  |  |     INDEXED BY messages_by_date_addressable | 
					
						
							|  |  |  |     WHERE | 
					
						
							|  |  |  |       conversationId IS ${conversationId} AND | 
					
						
							|  |  |  |       isAddressableMessage = 1 | 
					
						
							|  |  |  |     ORDER BY received_at DESC, sent_at DESC | 
					
						
							|  |  |  |     LIMIT ${limit}; | 
					
						
							|  |  |  |   `;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-12 14:45:54 -07:00
										 |  |  |   const rows = db.prepare(query).all<{ json: string }>(parameters); | 
					
						
							| 
									
										
										
										
											2025-01-16 13:34:35 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return rows.map(row => jsonToObject(row.json)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-29 01:56:00 +10:00
										 |  |  | function generateMessage(json: MessageAttributesType) { | 
					
						
							|  |  |  |   const { conversationId, received_at, sent_at, type } = json; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return { | 
					
						
							|  |  |  |     conversationId, | 
					
						
							|  |  |  |     json, | 
					
						
							|  |  |  |     received_at, | 
					
						
							|  |  |  |     sent_at, | 
					
						
							|  |  |  |     type, | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | describe('SQL/updateToSchemaVersion1060', () => { | 
					
						
							| 
									
										
										
										
											2024-07-22 11:16:33 -07:00
										 |  |  |   let db: WritableDB; | 
					
						
							| 
									
										
										
										
											2024-05-29 01:56:00 +10:00
										 |  |  |   beforeEach(() => { | 
					
						
							| 
									
										
										
										
											2024-07-22 11:16:33 -07:00
										 |  |  |     db = createDB(); | 
					
						
							| 
									
										
										
										
											2024-05-29 01:56:00 +10:00
										 |  |  |     updateToVersion(db, 1060); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   afterEach(() => { | 
					
						
							|  |  |  |     db.close(); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   describe('Addressable Messages', () => { | 
					
						
							|  |  |  |     describe('Storing of new attachment jobs', () => { | 
					
						
							|  |  |  |       it('returns only incoming/outgoing messages', () => { | 
					
						
							|  |  |  |         const conversationId = generateGuid(); | 
					
						
							|  |  |  |         const otherConversationId = generateGuid(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         insertData(db, 'messages', [ | 
					
						
							|  |  |  |           generateMessage({ | 
					
						
							|  |  |  |             id: '1', | 
					
						
							|  |  |  |             conversationId, | 
					
						
							|  |  |  |             type: 'incoming', | 
					
						
							|  |  |  |             received_at: 1, | 
					
						
							|  |  |  |             sent_at: 1, | 
					
						
							|  |  |  |             timestamp: 1, | 
					
						
							|  |  |  |           }), | 
					
						
							|  |  |  |           generateMessage({ | 
					
						
							|  |  |  |             id: '2', | 
					
						
							|  |  |  |             conversationId, | 
					
						
							|  |  |  |             type: 'story', | 
					
						
							|  |  |  |             received_at: 2, | 
					
						
							|  |  |  |             sent_at: 2, | 
					
						
							|  |  |  |             timestamp: 2, | 
					
						
							|  |  |  |           }), | 
					
						
							|  |  |  |           generateMessage({ | 
					
						
							|  |  |  |             id: '3', | 
					
						
							|  |  |  |             conversationId, | 
					
						
							|  |  |  |             type: 'outgoing', | 
					
						
							|  |  |  |             received_at: 3, | 
					
						
							|  |  |  |             sent_at: 3, | 
					
						
							|  |  |  |             timestamp: 3, | 
					
						
							|  |  |  |           }), | 
					
						
							|  |  |  |           generateMessage({ | 
					
						
							|  |  |  |             id: '4', | 
					
						
							|  |  |  |             conversationId, | 
					
						
							|  |  |  |             type: 'group-v1-migration', | 
					
						
							|  |  |  |             received_at: 4, | 
					
						
							|  |  |  |             sent_at: 4, | 
					
						
							|  |  |  |             timestamp: 4, | 
					
						
							|  |  |  |           }), | 
					
						
							|  |  |  |           generateMessage({ | 
					
						
							|  |  |  |             id: '5', | 
					
						
							|  |  |  |             conversationId, | 
					
						
							|  |  |  |             type: 'group-v2-change', | 
					
						
							|  |  |  |             received_at: 5, | 
					
						
							|  |  |  |             sent_at: 5, | 
					
						
							|  |  |  |             timestamp: 5, | 
					
						
							|  |  |  |           }), | 
					
						
							|  |  |  |           generateMessage({ | 
					
						
							|  |  |  |             id: '6', | 
					
						
							|  |  |  |             conversationId, | 
					
						
							|  |  |  |             type: 'incoming', | 
					
						
							|  |  |  |             received_at: 6, | 
					
						
							|  |  |  |             sent_at: 6, | 
					
						
							|  |  |  |             timestamp: 6, | 
					
						
							|  |  |  |           }), | 
					
						
							|  |  |  |           generateMessage({ | 
					
						
							|  |  |  |             id: '7', | 
					
						
							|  |  |  |             conversationId, | 
					
						
							|  |  |  |             type: 'profile-change', | 
					
						
							|  |  |  |             received_at: 7, | 
					
						
							|  |  |  |             sent_at: 7, | 
					
						
							|  |  |  |             timestamp: 7, | 
					
						
							|  |  |  |           }), | 
					
						
							|  |  |  |           generateMessage({ | 
					
						
							|  |  |  |             id: '8', | 
					
						
							|  |  |  |             conversationId: otherConversationId, | 
					
						
							|  |  |  |             type: 'incoming', | 
					
						
							|  |  |  |             received_at: 8, | 
					
						
							|  |  |  |             sent_at: 8, | 
					
						
							|  |  |  |             timestamp: 8, | 
					
						
							|  |  |  |           }), | 
					
						
							|  |  |  |         ]); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-22 11:16:33 -07:00
										 |  |  |         const messages = getMostRecentAddressableMessages(db, conversationId); | 
					
						
							| 
									
										
										
										
											2024-05-29 01:56:00 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  |         assert.lengthOf(messages, 3); | 
					
						
							|  |  |  |         assert.deepEqual(messages, [ | 
					
						
							|  |  |  |           { | 
					
						
							|  |  |  |             id: '6', | 
					
						
							|  |  |  |             conversationId, | 
					
						
							|  |  |  |             type: 'incoming', | 
					
						
							|  |  |  |             received_at: 6, | 
					
						
							|  |  |  |             sent_at: 6, | 
					
						
							|  |  |  |             timestamp: 6, | 
					
						
							|  |  |  |           }, | 
					
						
							|  |  |  |           { | 
					
						
							|  |  |  |             id: '3', | 
					
						
							|  |  |  |             conversationId, | 
					
						
							|  |  |  |             type: 'outgoing', | 
					
						
							|  |  |  |             received_at: 3, | 
					
						
							|  |  |  |             sent_at: 3, | 
					
						
							|  |  |  |             timestamp: 3, | 
					
						
							|  |  |  |           }, | 
					
						
							|  |  |  |           { | 
					
						
							|  |  |  |             id: '1', | 
					
						
							|  |  |  |             conversationId, | 
					
						
							|  |  |  |             type: 'incoming', | 
					
						
							|  |  |  |             received_at: 1, | 
					
						
							|  |  |  |             sent_at: 1, | 
					
						
							|  |  |  |             timestamp: 1, | 
					
						
							|  |  |  |           }, | 
					
						
							|  |  |  |         ]); | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-22 11:16:33 -07:00
										 |  |  |       it('ensures that index is used for getMostRecentAddressableMessages, with storyId', () => { | 
					
						
							| 
									
										
										
										
											2025-03-12 14:45:54 -07:00
										 |  |  |         const detail = explain( | 
					
						
							|  |  |  |           db, | 
					
						
							|  |  |  |           sql`
 | 
					
						
							| 
									
										
										
										
											2024-05-29 01:56:00 +10:00
										 |  |  |           SELECT json FROM messages | 
					
						
							|  |  |  |           INDEXED BY messages_by_date_addressable | 
					
						
							|  |  |  |           WHERE | 
					
						
							|  |  |  |             conversationId IS 'not-important' AND | 
					
						
							|  |  |  |             isAddressableMessage = 1 | 
					
						
							|  |  |  |           ORDER BY received_at DESC, sent_at DESC | 
					
						
							|  |  |  |           LIMIT 5; | 
					
						
							|  |  |  |           `
 | 
					
						
							| 
									
										
										
										
											2025-03-12 14:45:54 -07:00
										 |  |  |         ); | 
					
						
							| 
									
										
										
										
											2024-05-29 01:56:00 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  |         assert.notInclude(detail, 'B-TREE'); | 
					
						
							|  |  |  |         assert.notInclude(detail, 'SCAN'); | 
					
						
							|  |  |  |         assert.include( | 
					
						
							|  |  |  |           detail, | 
					
						
							|  |  |  |           'SEARCH messages USING INDEX messages_by_date_addressable (conversationId=? AND isAddressableMessage=?)' | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   describe('Sync Tasks', () => { | 
					
						
							|  |  |  |     it('creates tasks in bulk, and fetches all', () => { | 
					
						
							|  |  |  |       const now = Date.now(); | 
					
						
							|  |  |  |       const expected: Array<SyncTaskType> = [ | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |           id: generateGuid(), | 
					
						
							|  |  |  |           attempts: 1, | 
					
						
							|  |  |  |           createdAt: now + 1, | 
					
						
							|  |  |  |           data: { | 
					
						
							|  |  |  |             jsonField: 'one', | 
					
						
							|  |  |  |             data: 1, | 
					
						
							|  |  |  |           }, | 
					
						
							|  |  |  |           envelopeId: 'envelope-id-1', | 
					
						
							|  |  |  |           sentAt: 1, | 
					
						
							|  |  |  |           type: 'delete-conversation', | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |           id: generateGuid(), | 
					
						
							|  |  |  |           attempts: 2, | 
					
						
							|  |  |  |           createdAt: now + 2, | 
					
						
							|  |  |  |           data: { | 
					
						
							|  |  |  |             jsonField: 'two', | 
					
						
							|  |  |  |             data: 2, | 
					
						
							|  |  |  |           }, | 
					
						
							|  |  |  |           envelopeId: 'envelope-id-2', | 
					
						
							|  |  |  |           sentAt: 2, | 
					
						
							|  |  |  |           type: 'delete-conversation', | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |           id: generateGuid(), | 
					
						
							|  |  |  |           attempts: 3, | 
					
						
							|  |  |  |           createdAt: now + 3, | 
					
						
							|  |  |  |           data: { | 
					
						
							|  |  |  |             jsonField: 'three', | 
					
						
							|  |  |  |             data: 3, | 
					
						
							|  |  |  |           }, | 
					
						
							|  |  |  |           envelopeId: 'envelope-id-3', | 
					
						
							|  |  |  |           sentAt: 3, | 
					
						
							|  |  |  |           type: 'delete-conversation', | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |       ]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-22 11:16:33 -07:00
										 |  |  |       saveSyncTasks(db, expected); | 
					
						
							| 
									
										
										
										
											2024-05-29 01:56:00 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-25 09:18:42 -06:00
										 |  |  |       const actual = dequeueOldestSyncTasks(db, { previousRowId: null }); | 
					
						
							|  |  |  |       assert.deepEqual( | 
					
						
							|  |  |  |         expected.map(t => ({ ...t, attempts: t.attempts + 1 })), | 
					
						
							|  |  |  |         actual.tasks, | 
					
						
							|  |  |  |         'before delete' | 
					
						
							|  |  |  |       ); | 
					
						
							| 
									
										
										
										
											2024-05-29 01:56:00 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-22 11:16:33 -07:00
										 |  |  |       removeSyncTaskById(db, expected[1].id); | 
					
						
							| 
									
										
										
										
											2024-05-29 01:56:00 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-25 09:18:42 -06:00
										 |  |  |       const actualAfterDelete = dequeueOldestSyncTasks(db, { | 
					
						
							|  |  |  |         previousRowId: null, | 
					
						
							|  |  |  |         incrementAttempts: false, | 
					
						
							|  |  |  |       }); | 
					
						
							| 
									
										
										
										
											2024-05-29 01:56:00 +10:00
										 |  |  |       assert.deepEqual( | 
					
						
							|  |  |  |         [ | 
					
						
							|  |  |  |           { ...expected[0], attempts: 2 }, | 
					
						
							|  |  |  |           { ...expected[2], attempts: 4 }, | 
					
						
							|  |  |  |         ], | 
					
						
							| 
									
										
										
										
											2024-12-04 14:03:29 -08:00
										 |  |  |         actualAfterDelete.tasks, | 
					
						
							| 
									
										
										
										
											2024-05-29 01:56:00 +10:00
										 |  |  |         'after delete' | 
					
						
							|  |  |  |       ); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-12-04 14:03:29 -08:00
										 |  |  |     it('dequeueOldestSyncTasks expired tasks', () => { | 
					
						
							| 
									
										
										
										
											2024-05-29 01:56:00 +10:00
										 |  |  |       const now = Date.now(); | 
					
						
							|  |  |  |       const twoWeeksAgo = now - WEEK * 2; | 
					
						
							|  |  |  |       const expected: Array<SyncTaskType> = [ | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |           id: generateGuid(), | 
					
						
							|  |  |  |           attempts: MAX_SYNC_TASK_ATTEMPTS, | 
					
						
							|  |  |  |           createdAt: twoWeeksAgo, | 
					
						
							|  |  |  |           data: { | 
					
						
							|  |  |  |             jsonField: 'expired', | 
					
						
							|  |  |  |             data: 1, | 
					
						
							|  |  |  |           }, | 
					
						
							|  |  |  |           envelopeId: 'envelope-id-1', | 
					
						
							|  |  |  |           sentAt: 1, | 
					
						
							|  |  |  |           type: 'delete-conversation', | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |           id: generateGuid(), | 
					
						
							|  |  |  |           attempts: 2, | 
					
						
							|  |  |  |           createdAt: twoWeeksAgo, | 
					
						
							|  |  |  |           data: { | 
					
						
							|  |  |  |             jsonField: 'old-but-few-attemts', | 
					
						
							|  |  |  |             data: 2, | 
					
						
							|  |  |  |           }, | 
					
						
							|  |  |  |           envelopeId: 'envelope-id-2', | 
					
						
							|  |  |  |           sentAt: 2, | 
					
						
							|  |  |  |           type: 'delete-conversation', | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |           id: generateGuid(), | 
					
						
							|  |  |  |           attempts: MAX_SYNC_TASK_ATTEMPTS * 2, | 
					
						
							|  |  |  |           createdAt: now, | 
					
						
							|  |  |  |           data: { | 
					
						
							|  |  |  |             jsonField: 'new-but-many-attempts', | 
					
						
							|  |  |  |             data: 3, | 
					
						
							|  |  |  |           }, | 
					
						
							|  |  |  |           envelopeId: 'envelope-id-3', | 
					
						
							|  |  |  |           sentAt: 3, | 
					
						
							|  |  |  |           type: 'delete-conversation', | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |           id: generateGuid(), | 
					
						
							|  |  |  |           attempts: MAX_SYNC_TASK_ATTEMPTS - 1, | 
					
						
							|  |  |  |           createdAt: now + 1, | 
					
						
							|  |  |  |           data: { | 
					
						
							|  |  |  |             jsonField: 'new-and-fresh', | 
					
						
							|  |  |  |             data: 4, | 
					
						
							|  |  |  |           }, | 
					
						
							|  |  |  |           envelopeId: 'envelope-id-4', | 
					
						
							|  |  |  |           sentAt: 4, | 
					
						
							|  |  |  |           type: 'delete-conversation', | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |       ]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-22 11:16:33 -07:00
										 |  |  |       saveSyncTasks(db, expected); | 
					
						
							| 
									
										
										
										
											2024-05-29 01:56:00 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-25 09:18:42 -06:00
										 |  |  |       const actual = dequeueOldestSyncTasks(db, { previousRowId: null }); | 
					
						
							| 
									
										
										
										
											2024-05-29 01:56:00 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-12-04 14:03:29 -08:00
										 |  |  |       assert.lengthOf(actual.tasks, 3); | 
					
						
							| 
									
										
										
										
											2025-02-25 09:18:42 -06:00
										 |  |  |       assert.deepEqual( | 
					
						
							|  |  |  |         [ | 
					
						
							|  |  |  |           { ...expected[1], attempts: 3 }, | 
					
						
							|  |  |  |           { ...expected[2], attempts: 11 }, | 
					
						
							|  |  |  |           { ...expected[3], attempts: 5 }, | 
					
						
							|  |  |  |         ], | 
					
						
							|  |  |  |         actual.tasks | 
					
						
							|  |  |  |       ); | 
					
						
							| 
									
										
										
										
											2024-05-29 01:56:00 +10:00
										 |  |  |     }); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | }); |