Finally.cs 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081
  1. // Licensed to the .NET Foundation under one or more agreements.
  2. // The .NET Foundation licenses this file to you under the MIT License.
  3. // See the LICENSE file in the project root for more information.
  4. using System.Collections.Generic;
  5. using System.Threading;
  6. using System.Threading.Tasks;
  7. namespace System.Linq
  8. {
  9. public static partial class AsyncEnumerableEx
  10. {
  11. /// <summary>
  12. /// Invokes a specified action after the source async-enumerable sequence terminates gracefully or exceptionally.
  13. /// </summary>
  14. /// <typeparam name="TSource">The type of the elements in the source sequence.</typeparam>
  15. /// <param name="source">Source sequence.</param>
  16. /// <param name="finallyAction">Action to invoke after the source async-enumerable sequence terminates.</param>
  17. /// <returns>Source sequence with the action-invoking termination behavior applied.</returns>
  18. /// <exception cref="ArgumentNullException"><paramref name="source"/> or <paramref name="finallyAction"/> is null.</exception>
  19. public static IAsyncEnumerable<TSource> Finally<TSource>(this IAsyncEnumerable<TSource> source, Action finallyAction)
  20. {
  21. if (source == null)
  22. throw Error.ArgumentNull(nameof(source));
  23. if (finallyAction == null)
  24. throw Error.ArgumentNull(nameof(finallyAction));
  25. return Core(source, finallyAction);
  26. static async IAsyncEnumerable<TSource> Core(IAsyncEnumerable<TSource> source, Action finallyAction, [System.Runtime.CompilerServices.EnumeratorCancellation] CancellationToken cancellationToken = default)
  27. {
  28. try
  29. {
  30. await foreach (var item in source.WithCancellation(cancellationToken).ConfigureAwait(false))
  31. {
  32. yield return item;
  33. }
  34. }
  35. finally
  36. {
  37. finallyAction();
  38. }
  39. }
  40. }
  41. /// <summary>
  42. /// Invokes a specified asynchronous action after the source async-enumerable sequence terminates gracefully or exceptionally.
  43. /// </summary>
  44. /// <typeparam name="TSource">The type of the elements in the source sequence.</typeparam>
  45. /// <param name="source">Source sequence.</param>
  46. /// <param name="finallyAction">Action to invoke and await asynchronously after the source async-enumerable sequence terminates.</param>
  47. /// <returns>Source sequence with the action-invoking termination behavior applied.</returns>
  48. /// <exception cref="ArgumentNullException"><paramref name="source"/> or <paramref name="finallyAction"/> is null.</exception>
  49. public static IAsyncEnumerable<TSource> Finally<TSource>(this IAsyncEnumerable<TSource> source, Func<Task> finallyAction)
  50. {
  51. if (source == null)
  52. throw Error.ArgumentNull(nameof(source));
  53. if (finallyAction == null)
  54. throw Error.ArgumentNull(nameof(finallyAction));
  55. return Core(source, finallyAction);
  56. static async IAsyncEnumerable<TSource> Core(IAsyncEnumerable<TSource> source, Func<Task> finallyAction, [System.Runtime.CompilerServices.EnumeratorCancellation] CancellationToken cancellationToken = default)
  57. {
  58. try
  59. {
  60. await foreach (var item in source.WithCancellation(cancellationToken).ConfigureAwait(false))
  61. {
  62. yield return item;
  63. }
  64. }
  65. finally
  66. {
  67. // REVIEW: No cancellation support for finally action.
  68. await finallyAction().ConfigureAwait(false);
  69. }
  70. }
  71. }
  72. }
  73. }