Files
swift-mirror/test/Concurrency/Runtime/async_taskgroup_throw_recover.swift
2021-01-23 07:27:34 +09:00

87 lines
2.1 KiB
Swift

// RUN: %target-run-simple-swift(-Xfrontend -enable-experimental-concurrency) | %FileCheck %s --dump-input always
// REQUIRES: executable_test
// REQUIRES: concurrency
// REQUIRES: OS=macosx
// REQUIRES: CPU=x86_64
import Dispatch
struct Boom: Error {
}
struct IgnoredBoom: Error {
}
func one()
async -> Int {
1
}
func boom()
async throws -> Int {
throw Boom()
}
func test_taskGroup_throws() async {
do {
let got = try await Task.withGroup(resultType: Int.self) {
group async throws -> Int in
await group.add { await one() }
await group.add { try await boom() }
do {
while let r = try await group.next() {
print("next: \(r)")
}
} catch {
print("error caught in group: \(error)")
await group.add { () async -> Int in
print("task 3 (cancelled: \(await Task.isCancelled()))")
return 3
}
guard let got = try! await group.next() else {
print("task group failed to get 3 (:\(#line))")
return 0
}
print("task group next: \(got)")
if got == 1 {
// the previous 1 completed before the 3 we just submitted,
// we still want to see that three so let's await for it
guard let third = try! await group.next() else {
print("task group failed to get 3 (:\(#line))")
return got
}
print("task group returning normally: \(third)")
return third
} else {
print("task group returning normally: \(got)")
return got
}
}
fatalError("Should have thrown and handled inside the catch block")
}
// Optionally, we may get the first result back before the failure:
// COM: next: 1
// CHECK: error caught in group: Boom()
// CHECK: task 3 (cancelled: false)
// CHECK: task group returning normally: 3
// CHECK: got: 3
print("got: \(got)")
} catch {
print("rethrown: \(error)")
fatalError("Expected recovered result, but got error: \(error)")
}
}
runAsyncAndBlock(test_taskGroup_throws)