You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
session-ios/_SharedTestUtilities/Mock.swift

491 lines
17 KiB
Swift

// Copyright © 2022 Rangeproof Pty Ltd. All rights reserved.
import Foundation
import SessionUtilitiesKit
// MARK: - MockError
public enum MockError: Error {
case mockedData
}
// MARK: - Mock<T>
public class Mock<T> {
private let functionHandler: MockFunctionHandler
internal let functionConsumer: FunctionConsumer
// MARK: - Initialization
internal required init(
functionHandler: MockFunctionHandler? = nil,
initialSetup: ((Mock<T>) -> ())? = nil
) {
self.functionConsumer = FunctionConsumer()
self.functionHandler = (functionHandler ?? self.functionConsumer)
initialSetup?(self)
}
// MARK: - MockFunctionHandler
@discardableResult internal func mock<Output>(funcName: String = #function, args: [Any?] = [], untrackedArgs: [Any?] = []) -> Output {
return mock(funcName: funcName, checkArgs: args, actionArgs: args, untrackedArgs: untrackedArgs)
}
@discardableResult internal func mock<Output>(funcName: String = #function, checkArgs: [Any?], actionArgs: [Any?], untrackedArgs: [Any?]) -> Output {
return functionHandler.mock(
funcName,
parameterCount: checkArgs.count,
parameterSummary: summary(for: checkArgs),
allParameterSummaryCombinations: summaries(for: checkArgs),
actionArgs: actionArgs,
untrackedArgs: untrackedArgs
)
}
internal func mockNoReturn(funcName: String = #function, args: [Any?] = [], untrackedArgs: [Any?] = []) {
mockNoReturn(funcName: funcName, checkArgs: args, actionArgs: args, untrackedArgs: untrackedArgs)
}
internal func mockNoReturn(funcName: String = #function, checkArgs: [Any?], actionArgs: [Any?], untrackedArgs: [Any?]) {
functionHandler.mockNoReturn(
funcName,
parameterCount: checkArgs.count,
parameterSummary: summary(for: checkArgs),
allParameterSummaryCombinations: summaries(for: checkArgs),
actionArgs: actionArgs,
untrackedArgs: untrackedArgs
)
}
@discardableResult internal func mockThrowing<Output>(funcName: String = #function, args: [Any?] = [], untrackedArgs: [Any?] = []) throws -> Output {
return try mockThrowing(funcName: funcName, checkArgs: args, actionArgs: args, untrackedArgs: untrackedArgs)
}
@discardableResult internal func mockThrowing<Output>(funcName: String = #function, checkArgs: [Any?], actionArgs: [Any?], untrackedArgs: [Any?]) throws -> Output {
return try functionHandler.mockThrowing(
funcName,
parameterCount: checkArgs.count,
parameterSummary: summary(for: checkArgs),
allParameterSummaryCombinations: summaries(for: checkArgs),
actionArgs: actionArgs,
untrackedArgs: untrackedArgs
)
}
// MARK: - Functions
internal func reset() {
functionConsumer.trackCalls = true
functionConsumer.functionBuilders = []
functionConsumer.functionHandlers = [:]
functionConsumer.calls.mutate { $0 = [:] }
}
internal func when<R>(_ callBlock: @escaping (inout T) throws -> R) -> MockFunctionBuilder<T, R> {
let builder: MockFunctionBuilder<T, R> = MockFunctionBuilder(callBlock, mockInit: type(of: self).init)
functionConsumer.functionBuilders.append(builder.build)
return builder
}
// MARK: - Convenience
private func summaries(for argument: Any) -> [ParameterCombination] {
switch argument {
case let array as [Any]:
return array.allCombinations()
.map { ParameterCombination(count: $0.count, summary: summary(for: $0)) }
default: return [ParameterCombination(count: 1, summary: summary(for: argument))]
}
}
private func summary(for argument: Any) -> String {
switch argument {
case let string as String: return string
case let array as [Any]: return "[\(array.map { summary(for: $0) }.joined(separator: ", "))]"
case let dict as [String: Any]:
if dict.isEmpty { return "[:]" }
let sortedValues: [String] = dict
.map { key, value in "\(summary(for: key)):\(summary(for: value))" }
.sorted()
return "[\(sortedValues.joined(separator: ", "))]"
case let data as Data: return "Data(base64Encoded: \(data.base64EncodedString()))"
default: return String(reflecting: argument) // Default to the `debugDescription` if available
}
}
}
// MARK: - MockFunctionHandler
protocol MockFunctionHandler {
func mock<Output>(
_ functionName: String,
parameterCount: Int,
parameterSummary: String,
allParameterSummaryCombinations: [ParameterCombination],
actionArgs: [Any?],
untrackedArgs: [Any?]
) -> Output
func mockNoReturn(
_ functionName: String,
parameterCount: Int,
parameterSummary: String,
allParameterSummaryCombinations: [ParameterCombination],
actionArgs: [Any?],
untrackedArgs: [Any?]
)
func mockThrowing<Output>(
_ functionName: String,
parameterCount: Int,
parameterSummary: String,
allParameterSummaryCombinations: [ParameterCombination],
actionArgs: [Any?],
untrackedArgs: [Any?]
) throws -> Output
}
// MARK: - CallDetails
internal struct CallDetails: Equatable, Hashable {
let parameterSummary: String
let allParameterSummaryCombinations: [ParameterCombination]
}
// MARK: - ParameterCombination
internal struct ParameterCombination: Equatable, Hashable {
let count: Int
let summary: String
}
// MARK: - MockFunction
internal class MockFunction {
var name: String
var parameterCount: Int
var parameterSummary: String
var allParameterSummaryCombinations: [ParameterCombination]
var actions: [([Any?], [Any?]) -> Void]
var returnError: (any Error)?
var returnValue: Any?
init(
name: String,
parameterCount: Int,
parameterSummary: String,
allParameterSummaryCombinations: [ParameterCombination],
actions: [([Any?], [Any?]) -> Void],
returnError: (any Error)?,
returnValue: Any?
) {
self.name = name
self.parameterCount = parameterCount
self.parameterSummary = parameterSummary
self.allParameterSummaryCombinations = allParameterSummaryCombinations
self.actions = actions
self.returnError = returnError
self.returnValue = returnValue
}
}
// MARK: - MockFunctionBuilder
internal class MockFunctionBuilder<T, R>: MockFunctionHandler {
private let callBlock: (inout T) throws -> R
private let mockInit: (MockFunctionHandler?, ((Mock<T>) -> ())?) -> Mock<T>
private var functionName: String?
private var parameterCount: Int?
private var parameterSummary: String?
private var allParameterSummaryCombinations: [ParameterCombination]?
private var actions: [([Any?], [Any?]) -> Void] = []
private var returnValue: R?
internal var returnValueGenerator: ((String, Int, String, [ParameterCombination]) -> R?)?
private var returnError: Error?
// MARK: - Initialization
init(_ callBlock: @escaping (inout T) throws -> R, mockInit: @escaping (MockFunctionHandler?, ((Mock<T>) -> ())?) -> Mock<T>) {
self.callBlock = callBlock
self.mockInit = mockInit
}
// MARK: - Behaviours
/// Closure parameter is an array of arguments called by the function
@discardableResult func then(_ block: @escaping ([Any?]) -> Void) -> MockFunctionBuilder<T, R> {
actions.append({ args, _ in block(args) })
return self
}
/// Closure parameters are an array of arguments, followed by an array of "untracked" arguments called by the function
@discardableResult func then(_ block: @escaping ([Any?], [Any?]) -> Void) -> MockFunctionBuilder<T, R> {
actions.append(block)
return self
}
func thenReturn(_ value: R?) {
returnValue = value
}
func thenThrow(_ error: Error) {
returnError = error
}
// MARK: - MockFunctionHandler
func mock<Output>(
_ functionName: String,
parameterCount: Int,
parameterSummary: String,
allParameterSummaryCombinations: [ParameterCombination],
actionArgs: [Any?],
untrackedArgs: [Any?]
) -> Output {
self.functionName = functionName
self.parameterCount = parameterCount
self.parameterSummary = parameterSummary
self.allParameterSummaryCombinations = allParameterSummaryCombinations
let result: Any? = (
returnValue ??
returnValueGenerator?(functionName, parameterCount, parameterSummary, allParameterSummaryCombinations)
)
return (result as! Output)
}
func mockNoReturn(
_ functionName: String,
parameterCount: Int,
parameterSummary: String,
allParameterSummaryCombinations: [ParameterCombination],
actionArgs: [Any?],
untrackedArgs: [Any?]
) {
self.functionName = functionName
self.parameterCount = parameterCount
self.parameterSummary = parameterSummary
self.allParameterSummaryCombinations = allParameterSummaryCombinations
}
func mockThrowing<Output>(
_ functionName: String,
parameterCount: Int,
parameterSummary: String,
allParameterSummaryCombinations: [ParameterCombination],
actionArgs: [Any?],
untrackedArgs: [Any?]
) throws -> Output {
self.functionName = functionName
self.parameterCount = parameterCount
self.parameterSummary = parameterSummary
self.allParameterSummaryCombinations = allParameterSummaryCombinations
if let returnError: Error = returnError { throw returnError }
let anyResult: Any? = (
returnValue ??
returnValueGenerator?(functionName, parameterCount, parameterSummary, allParameterSummaryCombinations)
)
switch anyResult {
case let value as Output: return value
default: throw MockError.mockedData
}
}
// MARK: - Build
func build() throws -> MockFunction {
var completionMock = mockInit(self, nil) as! T
_ = try? callBlock(&completionMock)
guard
let name: String = functionName,
let parameterCount: Int = parameterCount,
let parameterSummary: String = parameterSummary,
let allParameterSummaryCombinations: [ParameterCombination] = allParameterSummaryCombinations
else { preconditionFailure("Attempted to build the MockFunction before it was called") }
return MockFunction(
name: name,
parameterCount: parameterCount,
parameterSummary: parameterSummary,
allParameterSummaryCombinations: allParameterSummaryCombinations,
actions: actions,
returnError: returnError,
returnValue: returnValue
)
}
}
// MARK: - FunctionConsumer
internal class FunctionConsumer: MockFunctionHandler {
struct Key: Equatable, Hashable {
let name: String
let paramCount: Int
}
var trackCalls: Bool = true
var functionBuilders: [() throws -> MockFunction?] = []
var functionHandlers: [Key: [String: MockFunction]] = [:]
var calls: Atomic<[Key: [CallDetails]]> = Atomic([:])
private func getExpectation(
_ functionName: String,
parameterCount: Int,
parameterSummary: String,
allParameterSummaryCombinations: [ParameterCombination],
actionArgs: [Any?],
untrackedArgs: [Any?]
) -> MockFunction {
let key: Key = Key(name: functionName, paramCount: parameterCount)
if !functionBuilders.isEmpty {
functionBuilders
.compactMap { try? $0() }
.forEach { function in
let key: Key = Key(name: function.name, paramCount: function.parameterCount)
var updatedHandlers: [String: MockFunction] = (functionHandlers[key] ?? [:])
// Add the actual 'parameterSummary' value for the handlers (override any
// existing entries
updatedHandlers[function.parameterSummary] = function
// Upsert entries for all remaining combinations (assume we want to
// overwrite any existing entries)
function.allParameterSummaryCombinations.forEach { combination in
updatedHandlers[combination.summary] = function
}
functionHandlers[key] = updatedHandlers
}
functionBuilders.removeAll()
}
guard let expectation: MockFunction = firstFunction(for: key, matchingParameterSummaryIfPossible: parameterSummary, allParameterSummaryCombinations: allParameterSummaryCombinations) else {
preconditionFailure("No expectations found for \(functionName)")
}
// Record the call so it can be validated later (assuming we are tracking calls)
if trackCalls {
calls.mutate {
$0[key] = ($0[key] ?? []).appending(
CallDetails(
parameterSummary: parameterSummary,
allParameterSummaryCombinations: allParameterSummaryCombinations
)
)
}
}
for action in expectation.actions {
action(actionArgs, untrackedArgs)
}
return expectation
}
func mock<Output>(
_ functionName: String,
parameterCount: Int,
parameterSummary: String,
allParameterSummaryCombinations: [ParameterCombination],
actionArgs: [Any?],
untrackedArgs: [Any?]
) -> Output {
let expectation: MockFunction = getExpectation(
functionName,
parameterCount: parameterCount,
parameterSummary: parameterSummary,
allParameterSummaryCombinations: allParameterSummaryCombinations,
actionArgs: actionArgs,
untrackedArgs: untrackedArgs
)
return (expectation.returnValue as! Output)
}
func mockNoReturn(
_ functionName: String,
parameterCount: Int,
parameterSummary: String,
allParameterSummaryCombinations: [ParameterCombination],
actionArgs: [Any?],
untrackedArgs: [Any?]
) {
_ = getExpectation(
functionName,
parameterCount: parameterCount,
parameterSummary: parameterSummary,
allParameterSummaryCombinations: allParameterSummaryCombinations,
actionArgs: actionArgs,
untrackedArgs: untrackedArgs
)
}
func mockThrowing<Output>(
_ functionName: String,
parameterCount: Int,
parameterSummary: String,
allParameterSummaryCombinations: [ParameterCombination],
actionArgs: [Any?],
untrackedArgs: [Any?]
) throws -> Output {
let expectation: MockFunction = getExpectation(
functionName,
parameterCount: parameterCount,
parameterSummary: parameterSummary,
allParameterSummaryCombinations: allParameterSummaryCombinations,
actionArgs: actionArgs,
untrackedArgs: untrackedArgs
)
switch (expectation.returnError, expectation.returnValue) {
case (.some(let error), _): throw error
case (_, .some(let value as Output)): return value
default: throw MockError.mockedData
}
}
func firstFunction(
for key: Key,
matchingParameterSummaryIfPossible parameterSummary: String,
allParameterSummaryCombinations: [ParameterCombination]
) -> MockFunction? {
guard let possibleExpectations: [String: MockFunction] = functionHandlers[key] else { return nil }
guard let expectation: MockFunction = possibleExpectations[parameterSummary] else {
// We didn't find an exact match so try to find the match with the most matching parameters,
// do this by sorting based on the largest param count and checking if there is a match
let maybeExpectation: MockFunction? = allParameterSummaryCombinations
.sorted(by: { lhs, rhs in lhs.count > rhs.count })
.compactMap { combination in possibleExpectations[combination.summary] }
.first
if let expectation: MockFunction = maybeExpectation {
return expectation
}
// A `nil` response might be value but in a lot of places we will need to force-cast
// so try to find a non-nil response first
return (
possibleExpectations.values.first(where: { $0.returnValue != nil }) ??
possibleExpectations.values.first
)
}
return expectation
}
}