Skip to content

Commit 8d19029

Browse files
Merge pull request #2 from MetalheadSanya/property-verify
Add codegeneration to macro for verifying property getting and setting
2 parents 18e81e0 + e8272cb commit 8d19029

File tree

83 files changed

+434
-163
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

83 files changed

+434
-163
lines changed

Sources/SwiftMock/Documentation.docc/Usage/Verifying.md

+54-1
Original file line numberDiff line numberDiff line change
@@ -31,9 +31,61 @@ func test() {
3131

3232
By default, verify checks that the method was called exactly one time. If the method was called more than once, you will receive an error.
3333

34+
### Checking that property was read
35+
36+
To verify that the mock property has been read, use the ``verify(_:times:)`` method and pass the mock object as the first argument. This method will create a `Verify` structure for you that has a method to verify that your property has been read. The name of the method is `propertyGetter()`, where `property` is the name of your property. All rules regarding method call verification work similarly for properties.
37+
38+
```swift
39+
@Mock
40+
public protocol SomeService {
41+
var property: Int { get }
42+
}
43+
44+
45+
func test() {
46+
let mock = SomeServiceMock()
47+
48+
when(mock.$propertyGetter())
49+
.thenReturn(4)
50+
51+
_ = mock.property
52+
53+
verify(mock).propertyGetter()
54+
}
55+
```
56+
57+
By default, verify checks that the property was read exactly one time. If the property was read more than once, you will receive an error.
58+
59+
### Checking that property was write
60+
61+
To verify that the mock property has been write, use the ``verify(_:times:)`` method and pass the mock object as the first argument. This method will create a `Verify` structure for you that has a method to verify that your property has been write. The name of the method is `propertySetter(_:)`, where `property` is the name of your property. All rules regarding method call verification work similarly for properties. Additionally, you can pass any `AttributeMatcher` to the `propertySetter(_:)` method to check that a certain value has been set to the property.
62+
63+
```swift
64+
@Mock
65+
public protocol SomeService {
66+
var property: Int { get set }
67+
}
68+
69+
70+
func test() {
71+
let mock = SomeServiceMock()
72+
73+
when(mock.$propertySetter())
74+
.thenReturn()
75+
76+
mock.property = 6
77+
78+
// Checking that any value has been set to the property.
79+
verify(mock).propertySetter()
80+
81+
// Checking that 6 (six) has been set to the property.
82+
verify(mock).propertySetter(eq(6))
83+
}
84+
```
85+
3486
### Checking that method was called with specific arguments
3587

36-
The methods of a `Verify` structure have all the same methods as your protocol with one exception. All method arguments are replaced with a wrapper in the form of ``ArgumentMatcher``. As with stubbing, the default argument value is ``any()``. If you want to check that a method was called with a certain argument, then you need to specify the appropriate ``ArgumentMatcher`` for the arguments you want to check.
88+
The methods of a `Verify` structure have all the same methods as your protocol with one exception. All method arguments are replaced with a wrapper in the form of `ArgumentMatcher`. As with stubbing, the default argument value is ``any()``. If you want to check that a method was called with a certain argument, then you need to specify the appropriate `ArgumentMatcher` for the arguments you want to check.
3789

3890
```swift
3991
@Mock
@@ -55,6 +107,7 @@ func test() {
55107
verify(mock).getAlbumName(id: eq("id2"))
56108
}
57109
```
110+
58111
### Checking that method was called specific times
59112

60113
The ``verify(_:times:)`` method can check not only that the method was called once, but also a specific number of times. The `times` parameter of the ``verify(_:times:)`` method is used for this. This parameter type is ``TimesMatcher`` and you can use ``times(_:)``, ``never()``, ``atLeast(_:)``, ``atMost(_:)`` as examples, or create your own.

Sources/SwiftMockMacros/General.swift

+33-20
Original file line numberDiff line numberDiff line change
@@ -34,6 +34,19 @@ extension MockMacro {
3434
)
3535
}
3636

37+
static func makeMethodCallType(
38+
arguments: [TypeSyntax]
39+
) -> TypeSyntax {
40+
TypeSyntax(
41+
fromProtocol: IdentifierTypeSyntax(
42+
name: .identifier("MethodCall"),
43+
genericArgumentClause: GenericArgumentClauseSyntax {
44+
GenericArgumentSyntax(argument: makeTupleType(from: arguments))
45+
}
46+
)
47+
)
48+
}
49+
3750
private static func makeMethodWrapperType(
3851
baseName: TokenSyntax,
3952
isAsync: Bool,
@@ -114,33 +127,33 @@ extension MockMacro {
114127
private static func makeTupleType<T: Collection>(
115128
from types: T
116129
) -> TypeSyntax where T.Element == TypeSyntax {
117-
func packParametersToTupleType<Z: Collection>(
118-
_ types: Z
119-
) -> TupleTypeElementListSyntax where Z.Element == TypeSyntax {
120-
if types.count <= 1 {
121-
return TupleTypeElementListSyntax {
122-
for type in types {
123-
TupleTypeElementSyntax(type: type)
124-
}
125-
}
126-
} else {
127-
let rest = types.dropFirst()
128-
return TupleTypeElementListSyntax {
129-
TupleTypeElementSyntax(type: types.first!)
130-
TupleTypeElementSyntax(
131-
type: TupleTypeSyntax(elements: packParametersToTupleType(rest))
132-
)
133-
}
134-
}
135-
}
136-
137130
return TypeSyntax(
138131
fromProtocol: TupleTypeSyntax(
139132
elements: packParametersToTupleType(types)
140133
)
141134
)
142135
}
143136

137+
private static func packParametersToTupleType<Z: Collection>(
138+
_ types: Z
139+
) -> TupleTypeElementListSyntax where Z.Element == TypeSyntax {
140+
if types.count <= 1 {
141+
return TupleTypeElementListSyntax {
142+
for type in types {
143+
TupleTypeElementSyntax(type: type)
144+
}
145+
}
146+
} else {
147+
let rest = types.dropFirst()
148+
return TupleTypeElementListSyntax {
149+
TupleTypeElementSyntax(type: types.first!)
150+
TupleTypeElementSyntax(
151+
type: TupleTypeSyntax(elements: packParametersToTupleType(rest))
152+
)
153+
}
154+
}
155+
}
156+
144157
// MARK: - Making Labeled Expressions
145158

146159
static func makeMethodSignatureRegisterLabeledExpr(from containerToken: TokenSyntax) -> LabeledExprSyntax {

Sources/SwiftMockMacros/Property.swift

+12-9
Original file line numberDiff line numberDiff line change
@@ -5,20 +5,21 @@ extension MockMacro {
55

66
static func makeVariableMock(from variableDecl: VariableDeclSyntax, mockTypeToken: TokenSyntax) -> [DeclSyntax] {
77
var declarations: [DeclSyntax] = []
8-
for binding in variableDecl.bindings {
9-
guard let accessorBlock = binding.accessorBlock else {
8+
for bindingSyntax in variableDecl.bindings {
9+
guard let accessorBlock = bindingSyntax.accessorBlock else {
1010
// TODO: assertion
1111
continue
1212
}
1313
guard case let .`accessors`(accessorList) = accessorBlock.accessors else {
1414
// TODO: assertion
1515
continue
1616
}
17-
for accessor in accessorList {
18-
declarations.append(makeInvocationContainerProperty(patternBinding: binding, accessorDecl: accessor))
19-
declarations.append(makeSignatureMethod(patternBinding: binding, accessorDecl: accessor))
17+
for accessorDecl in accessorList {
18+
declarations.append(makeInvocationContainerProperty(patternBinding: bindingSyntax, accessorDecl: accessorDecl))
19+
declarations.append(makeCallStorageProperty(bindingSyntax: bindingSyntax, accessorDecl: accessorDecl))
20+
declarations.append(makeSignatureMethod(patternBinding: bindingSyntax, accessorDecl: accessorDecl))
2021
}
21-
declarations.append(makeMockProperty(bindingSyntax: binding, mockTypeToken: mockTypeToken))
22+
declarations.append(makeMockProperty(bindingSyntax: bindingSyntax, mockTypeToken: mockTypeToken))
2223
}
2324
return declarations
2425
}
@@ -83,7 +84,7 @@ extension MockMacro {
8384
)
8485
}
8586

86-
private static func makeGetterInvocationContainerToken(from bindingSyntax: PatternBindingSyntax) -> TokenSyntax {
87+
static func makeGetterInvocationContainerToken(from bindingSyntax: PatternBindingSyntax) -> TokenSyntax {
8788
let propertyPattern = bindingSyntax.pattern.as(IdentifierPatternSyntax.self) ?? IdentifierPatternSyntax(identifier: .identifier("unknown"))
8889
return .identifier(propertyPattern.identifier.text + "___getter")
8990
}
@@ -118,7 +119,7 @@ extension MockMacro {
118119
)
119120
}
120121

121-
private static func makeSetterInvocationContainerToken(from bindingSyntax: PatternBindingSyntax) -> TokenSyntax {
122+
static func makeSetterInvocationContainerToken(from bindingSyntax: PatternBindingSyntax) -> TokenSyntax {
122123
let propertyPattern = bindingSyntax.pattern.as(IdentifierPatternSyntax.self) ?? IdentifierPatternSyntax(identifier: .identifier("unknown"))
123124
return .identifier(propertyPattern.identifier.text + "___setter")
124125
}
@@ -262,6 +263,7 @@ extension MockMacro {
262263
accessorSpecifier: .keyword(.get)
263264
) {
264265
"let arguments = ()"
266+
makeStoreCallToStorageExpr(bindingSyntax: bindingSyntax, accessorDecl: AccessorDeclSyntax(accessorSpecifier: .keyword(.get)))
265267
ReturnStmtSyntax(
266268
expression: makeMockGetterReturnExpr(bindingSyntax: bindingSyntax, mockTypeToken: mockTypeToken)
267269
)
@@ -282,6 +284,7 @@ extension MockMacro {
282284
accessorSpecifier: .keyword(.set)
283285
) {
284286
"let arguments = (newValue)"
287+
makeStoreCallToStorageExpr(bindingSyntax: bindingSyntax, accessorDecl: AccessorDeclSyntax(accessorSpecifier: .keyword(.set)))
285288
ReturnStmtSyntax(
286289
expression: makeMockSetterReturnExpr(bindingSyntax: bindingSyntax, mockTypeToken: mockTypeToken)
287290
)
@@ -365,7 +368,7 @@ extension MockMacro {
365368

366369
// MARK: - General functions
367370

368-
private static func getBindingType(from bindingSyntax: PatternBindingSyntax) -> TypeSyntax {
371+
static func getBindingType(from bindingSyntax: PatternBindingSyntax) -> TypeSyntax {
369372
// TODO: error of unknown type
370373
bindingSyntax.typeAnnotation?.type.trimmed ?? voidType
371374
}

0 commit comments

Comments
 (0)