Signal-iOS/Signal/test/PerformanceTests/MessageProcessingPerformanceTest.swift
2020-06-18 09:28:49 -03:00

168 lines
5.3 KiB
Swift

//
// Copyright (c) 2020 Open Whisper Systems. All rights reserved.
//
import XCTest
@testable import SignalServiceKit
import GRDB
class MessageProcessingPerformanceTest: PerformanceBaseTest {
// MARK: - Dependencies
var messageReceiver: OWSMessageReceiver {
return SSKEnvironment.shared.messageReceiver
}
var tsAccountManager: TSAccountManager {
return SSKEnvironment.shared.tsAccountManager
}
var identityManager: OWSIdentityManager {
return SSKEnvironment.shared.identityManager
}
// MARK: -
let localE164Identifier = "+13235551234"
let localUUID = UUID()
let aliceE164Identifier = "+14715355555"
var aliceClient: SignalClient!
let bobE164Identifier = "+18083235555"
var bobClient: SignalClient!
let localClient = LocalSignalClient()
let runner = TestProtocolRunner()
lazy var fakeService = FakeService(localClient: localClient, runner: runner)
var dbObserverBlock: (() -> Void)?
private var dbObserver: BlockObserver?
// MARK: - Hooks
override func setUp() {
super.setUp()
// for unit tests, we must manually start the decryptJobQueue
SSKEnvironment.shared.messageDecryptJobQueue.setup()
let dbObserver = BlockObserver(block: { [weak self] in self?.dbObserverBlock?() })
self.dbObserver = dbObserver
databaseStorage.appendUIDatabaseSnapshotDelegate(dbObserver)
}
override func tearDown() {
super.tearDown()
self.dbObserver = nil
}
// MARK: - Tests
func testGRDBPerf_messageProcessing() {
storageCoordinator.useGRDBForTests()
try! databaseStorage.grdbStorage.setupUIDatabase()
measureMetrics(XCTestCase.defaultPerformanceMetrics, automaticallyStartMeasuring: false) {
processIncomingMessages()
}
databaseStorage.grdbStorage.testing_tearDownUIDatabase()
}
func testYapDBPerf_messageProcessing() {
// Getting this working will require a new observer pattern.
storageCoordinator.useYDBForTests()
measureMetrics(XCTestCase.defaultPerformanceMetrics, automaticallyStartMeasuring: false) {
processIncomingMessages()
}
}
func processIncomingMessages() {
// ensure local client has necessary "registered" state
identityManager.generateNewIdentityKey()
tsAccountManager.registerForTests(withLocalNumber: localE164Identifier, uuid: localUUID)
// use the uiDatabase to be notified of DB writes so we can verify the expected
// changes occur
bobClient = FakeSignalClient.generate(e164Identifier: bobE164Identifier)
aliceClient = FakeSignalClient.generate(e164Identifier: aliceE164Identifier)
write { transaction in
XCTAssertEqual(0, TSMessage.anyCount(transaction: transaction))
XCTAssertEqual(0, TSThread.anyCount(transaction: transaction))
try! self.runner.initialize(senderClient: self.bobClient,
recipientClient: self.localClient,
transaction: transaction)
}
let buildEnvelopeData = { () -> Data in
let envelopeBuilder = try! self.fakeService.envelopeBuilder(fromSenderClient: self.bobClient)
envelopeBuilder.setSourceE164(self.bobClient.e164Identifier!)
return try! envelopeBuilder.buildSerializedData()
}
let envelopeDatas: [Data] = (0..<500).map { _ in buildEnvelopeData() }
let expectMessagesProcessed = expectation(description: "messages processed")
var hasFulfilled = false
let fulfillOnce = {
if !hasFulfilled {
hasFulfilled = true
expectMessagesProcessed.fulfill()
}
}
self.dbObserverBlock = {
let messageCount = self.databaseStorage.read { transaction in
return TSInteraction.anyCount(transaction: transaction)
}
if messageCount == envelopeDatas.count {
fulfillOnce()
}
}
startMeasuring()
for envelopeData in envelopeDatas {
messageReceiver.handleReceivedEnvelopeData(envelopeData)
}
waitForExpectations(timeout: 15.0) { _ in
self.stopMeasuring()
self.dbObserverBlock = nil
self.write { transaction in
TSInteraction.anyRemoveAllWithInstantation(transaction: transaction)
TSThread.anyRemoveAllWithInstantation(transaction: transaction)
SSKMessageDecryptJobRecord.anyRemoveAllWithInstantation(transaction: transaction)
OWSMessageContentJob.anyRemoveAllWithInstantation(transaction: transaction)
OWSRecipientIdentity.anyRemoveAllWithInstantation(transaction: transaction)
}
}
}
}
private class BlockObserver: UIDatabaseSnapshotDelegate {
let block: () -> Void
init(block: @escaping () -> Void) {
self.block = block
}
func uiDatabaseSnapshotWillUpdate() {
AssertIsOnMainThread()
}
func uiDatabaseSnapshotDidUpdate(databaseChanges: UIDatabaseChanges) {
block()
}
func uiDatabaseSnapshotDidUpdateExternally() {
block()
}
func uiDatabaseSnapshotDidReset() {
block()
}
}