| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980 | using System;using System.Diagnostics;using System.Threading;using System.Threading.Tasks;namespace MediaBrowser.Common.Extensions{    /// <summary>    /// Extension methods for <see cref="Process"/>.    /// </summary>    public static class ProcessExtensions    {        /// <summary>        /// Asynchronously wait for the process to exit.        /// </summary>        /// <param name="process">The process to wait for.</param>        /// <param name="timeout">The duration to wait before cancelling waiting for the task.</param>        /// <returns>True if the task exited normally, false if the timeout elapsed before the process exited.</returns>        /// <exception cref="InvalidOperationException">If <see cref="Process.EnableRaisingEvents"/> is not set to true for the process.</exception>        public static async Task<bool> WaitForExitAsync(this Process process, TimeSpan timeout)        {            using (var cancelTokenSource = new CancellationTokenSource(timeout))            {                return await WaitForExitAsync(process, cancelTokenSource.Token).ConfigureAwait(false);            }        }        /// <summary>        /// Asynchronously wait for the process to exit.        /// </summary>        /// <param name="process">The process to wait for.</param>        /// <param name="cancelToken">A <see cref="CancellationToken"/> to observe while waiting for the process to exit.</param>        /// <returns>True if the task exited normally, false if cancelled before the process exited.</returns>        public static async Task<bool> WaitForExitAsync(this Process process, CancellationToken cancelToken)        {            if (!process.EnableRaisingEvents)            {                throw new InvalidOperationException("EnableRisingEvents must be enabled to async wait for a task to exit.");            }            // Add an event handler for the process exit event            var tcs = new TaskCompletionSource<bool>();            process.Exited += (sender, args) => tcs.TrySetResult(true);            // Return immediately if the process has already exited            if (process.HasExitedSafe())            {                return true;            }            // Register with the cancellation token then await            using (var cancelRegistration = cancelToken.Register(() => tcs.TrySetResult(process.HasExitedSafe())))            {                return await tcs.Task.ConfigureAwait(false);            }        }        /// <summary>        /// Gets a value indicating whether the associated process has been terminated using        /// <see cref="Process.HasExited"/>. This is safe to call even if there is no operating system process        /// associated with the <see cref="Process"/>.        /// </summary>        /// <param name="process">The process to check the exit status for.</param>        /// <returns>        /// True if the operating system process referenced by the <see cref="Process"/> component has        /// terminated, or if there is no associated operating system process; otherwise, false.        /// </returns>        private static bool HasExitedSafe(this Process process)        {            try            {                return process.HasExited;            }            catch (InvalidOperationException)            {                return true;            }        }    }}
 |