Swift Combine Sink Array AnyPublisher

I would like to wait for data from all elements and combine them into one result:

Item: AnyPublisher <Int, Swift.Error>
Array: AnyPublisher <[Result<Int, Swift.Error>], Never>

Can this be done somehow? I've tried using Zip and Merge - but I couldn't get the desired result.

Example:

func createItem(num: Int) -> AnyPublisher<Int, Swift.Error> {
    Just(num)
        .setFailureType(to: Swift.Error.self)
        .eraseToAnyPublisher()
}
func createItems(nums: [Int]) -> AnyPublisher<[Result<Int, Swift.Error>], Never> {
    Publishers.MergeMany(nums.map { self.createItem(num: $0) } )
        .collect()
        .eraseToAnyPublisher()
}

Function but "createItems" does not work

Upvotes: 0

Views: 2965

Answers (1)

New Dev
New Dev

Reputation: 49590

Looks like you want to generate an array of results - whether a value or an error - emitted by each publisher for each array value.

MergeMany + Collect is the right approach here. MergeMany will complete only when all its input publishers complete, and .collect() will for for that completion event before emitting.

Using your example:

func createItems(nums: [Int]) -> AnyPublisher<[Result<Int, Error>], Never> {

    let publishers = nums.map { v -> AnyPublisher<Result<Int, Error>, Never> in
        createItem(num: v) // AnyPublisher<Int, Error>
           .first() // ensure a single result
           .map { .success($0) } // map value to .success result
           .catch { Just(.failure($0)) } // map error to .failure result
           .eraseToAnyPublisher()
    }

    return Publishers.MergeMany(publishers)
        .collect() // wait for MergeMany to complete
        .eraseToAnyPublisher()
}

Upvotes: 1

Related Questions