Overall use case is 4 network requests
- A. Request 1 and 2 need to be in parallel and wait for both to complete
- B. Request 3 happens after 1 and 2 complete
- C. Request 4 happens after 3 completes
After C the final output should be of a AnyPublisher
type
I am trying to use Combine to achieve the above and so far I am able to do A using Publishers.Zip
and C using flatMap
. What I am struggling with is B. I can do that using nested completion handlers but not using flatMap
This is the sample code with Strings (not actual code)
func overallFunc(arg1: "arg1", arg2: "arg2", arg3: "arg3", arg4: "arg4" ) -> AnyPublisher<String?, Error> {
let pub1 = func1(arg1: arg1, arg2: arg2)
let pub2 = func2(arg1: arg3, arg2: arg4)
let combinedPub = Publishers.Zip(pub1, pub2)
combinedPub
.flatMap {(response1, response2) in
return func3(arg1: response1.attribute1, arg2: response2.attribute2)
}
}
The func1
, func2
and func3
all return URLSession.shared.dataTaskPublisher
with return type say AnyPublisher<String?, Error>
Now I am struggling with completing the code for overallFunc
. The complier gives the following error around flatMap
.
Type of expression is ambiguous without more context
If I add an extra return in the last line of overallFunc
then the error changes to
No 'flatMap' candidates produce the expected contextual result type 'AnyPublisher<String?, Error>'
In short I want to use flatMap
on the result of Publishers.Zip
and return another publisher so that I can add another flatMap
to do the 4th request but am not able to figure out the right syntax and order of things to do that.
CodePudding user response:
The result of flatMap
is going to be a String?
(the result of request 3) that you then want to pass to map
. That map
will convert the String into the fourth request. You could then add a subscriber to the fourth request which will be the result of your overall sequence. So...
pub1
.zip(pub2)
.flatMap { (response1, response2) in
func3(arg1: response1.attribute1, arg2: response2.attribute2)
}
.map {
(func3Result : String?) in
/* return publisher of 4th request */
}
.eraseToAnyPublisher()
The result of all of this will be the publisher of the 4th request.
CodePudding user response:
I think your attempt to obfuscate the code has removed the problem you were having. The following code compiles just fine:
func func1(arg1: String, arg2: String) -> AnyPublisher<String?, Error> { fatalError() }
func func2(arg1: String, arg2: String) -> AnyPublisher<String?, Error> { fatalError() }
func func3(arg1: String, arg2: String) -> AnyPublisher<String?, Error> { fatalError() }
func func4(arg1: String) -> AnyPublisher<String?, Error> { fatalError() }
func overallFunc(arg1: String, arg2: String, arg3: String, arg4: String) -> AnyPublisher<String?, Error> {
Publishers.Zip(
func1(arg1: arg1, arg2: arg2),
func2(arg1: arg3, arg2: arg4)
)
.flatMap { (response1, response2) in
func3(arg1: response1 ?? "", arg2: response2 ?? "")
}
.flatMap { response3 in
func4(arg1: response3 ?? "")
}
.eraseToAnyPublisher()
}