From b0a2c68ee60bd3d15889fa41cf1879c276a9e1b8 Mon Sep 17 00:00:00 2001 From: sr55 Date: Fri, 10 Apr 2020 20:28:16 +0100 Subject: [PATCH] WinGui: - Adding multi-instance support in the UI for the worker process. - Handling port conflicts better. It will now try up to 100 ports from the default port set in preferences. --- win/CS/HandBrake.Worker/Program.cs | 7 ++-- .../HandBrakeWPF/Instance/RemoteInstance.cs | 32 ++++++++++++++++--- .../Properties/Resources.Designer.cs | 2 +- win/CS/HandBrakeWPF/Properties/Resources.resx | 2 +- .../HandBrakeWPF/Services/Encode/LibEncode.cs | 23 +++++++++++-- .../Services/UserSettingService.cs | 2 +- 6 files changed, 55 insertions(+), 13 deletions(-) diff --git a/win/CS/HandBrake.Worker/Program.cs b/win/CS/HandBrake.Worker/Program.cs index 74114bd9ea73..f925ade38a60 100644 --- a/win/CS/HandBrake.Worker/Program.cs +++ b/win/CS/HandBrake.Worker/Program.cs @@ -41,17 +41,16 @@ public static void Main(string[] args) } } - Console.WriteLine("Starting HandBrake Engine ..."); + Console.WriteLine("Worker: Starting HandBrake Engine ..."); router = new ApiRouter(); router.TerminationEvent += Router_TerminationEvent; - Console.WriteLine("Starting Web Server ..."); - Console.WriteLine("Using Port: {0}", port); + Console.WriteLine("Worker: Starting Web Server on port {0} ...", port); Dictionary> apiHandlers = RegisterApiHandlers(); HttpServer webServer = new HttpServer(apiHandlers, port); webServer.Run(); - Console.WriteLine("Web Server Started"); + Console.WriteLine("Worker: Server Started"); manualResetEvent.WaitOne(); diff --git a/win/CS/HandBrakeWPF/Instance/RemoteInstance.cs b/win/CS/HandBrakeWPF/Instance/RemoteInstance.cs index 847e9cf4c338..1d11c1ccecfc 100644 --- a/win/CS/HandBrakeWPF/Instance/RemoteInstance.cs +++ b/win/CS/HandBrakeWPF/Instance/RemoteInstance.cs @@ -11,8 +11,12 @@ namespace HandBrakeWPF.Instance { using System; + using System.Collections.Generic; using System.Diagnostics; using System.IO; + using System.Linq; + using System.Net; + using System.Net.NetworkInformation; using System.Threading.Tasks; using System.Timers; using System.Windows.Media.Animation; @@ -46,7 +50,7 @@ public class RemoteInstance : HttpRequestBase, IEncodeInstance, IDisposable private readonly ILog logService; - private const double EncodePollIntervalMs = 250; + private const double EncodePollIntervalMs = 500; private Process workerProcess; private Timer encodePollTimer; @@ -56,7 +60,7 @@ public RemoteInstance(HBConfiguration configuration, ILog logService) { this.configuration = configuration; this.logService = logService; - this.port = configuration.RemoteServicePort; + this.port = this.GetOpenPort(this.configuration.RemoteServicePort); this.serverUrl = string.Format("http://127.0.0.1:{0}/", this.port); } @@ -139,7 +143,7 @@ private async void StartServer() StartInfo = { FileName = "HandBrake.Worker.exe", - Arguments = string.Format(" --port={0}", this.port), + Arguments = string.Format(" --port={0}", port), UseShellExecute = false, RedirectStandardOutput = true, RedirectStandardError = true, @@ -155,7 +159,7 @@ private async void StartServer() workerProcess.BeginErrorReadLine(); - this.logService.LogMessage(string.Format("Worker Process started with Process ID: {0}", this.workerProcess.Id)); + this.logService.LogMessage(string.Format("Worker Process started with Process ID: {0} and port: {1}", this.workerProcess.Id, port)); } } @@ -267,5 +271,25 @@ private async void PollEncodeProgress() this.EncodeCompleted?.Invoke(sender: this, e: new EncodeCompletedEventArgs(state.WorkDone.Error)); } } + + private int GetOpenPort(int startPort) + { + int portStartIndex = startPort; + + IPGlobalProperties properties = IPGlobalProperties.GetIPGlobalProperties(); + IPEndPoint[] tcpEndPoints = properties.GetActiveTcpListeners(); + + List usedPorts = tcpEndPoints.Select(p => p.Port).ToList(); + int unusedPort = 0; + + unusedPort = Enumerable.Range(portStartIndex, 99).FirstOrDefault(p => !usedPorts.Contains(p)); + + if (startPort != unusedPort) + { + this.logService.LogMessage(string.Format("Port {0} in use. Using {1} instead", startPort, unusedPort)); + } + + return unusedPort; + } } } diff --git a/win/CS/HandBrakeWPF/Properties/Resources.Designer.cs b/win/CS/HandBrakeWPF/Properties/Resources.Designer.cs index d89147d4f6aa..be72f0b9dd7c 100644 --- a/win/CS/HandBrakeWPF/Properties/Resources.Designer.cs +++ b/win/CS/HandBrakeWPF/Properties/Resources.Designer.cs @@ -3447,7 +3447,7 @@ public static string OptionsView_EnableVceEncoding { } /// - /// Looks up a localized string similar to Run each queued job in a separate worker process. (Experimental! Note, Limited to one instance of HandBrake currently!). + /// Looks up a localized string similar to Run each queued job in a separate worker process. (Experimental). /// public static string OptionsView_EnableWorkerProcesses { get { diff --git a/win/CS/HandBrakeWPF/Properties/Resources.resx b/win/CS/HandBrakeWPF/Properties/Resources.resx index f1b86d39e684..7acca6007dd3 100644 --- a/win/CS/HandBrakeWPF/Properties/Resources.resx +++ b/win/CS/HandBrakeWPF/Properties/Resources.resx @@ -2210,7 +2210,7 @@ This will also stop any existing running jobs. Please select a single job to view summary information. - Run each queued job in a separate worker process. (Experimental! Note, Limited to one instance of HandBrake currently!) + Run each queued job in a separate worker process. (Experimental) Default network port for worker: diff --git a/win/CS/HandBrakeWPF/Services/Encode/LibEncode.cs b/win/CS/HandBrakeWPF/Services/Encode/LibEncode.cs index c59326f4eeb9..02e5dc2c4b54 100644 --- a/win/CS/HandBrakeWPF/Services/Encode/LibEncode.cs +++ b/win/CS/HandBrakeWPF/Services/Encode/LibEncode.cs @@ -200,7 +200,7 @@ public EncodeTask GetActiveJob() /// Log message content protected void ServiceLogMessage(string message) { - this.log.LogMessage(string.Format("{0}# {1}{0}", Environment.NewLine, message)); + this.log.LogMessage(string.Format("{0}# {1}", Environment.NewLine, message)); } protected void TimedLogMessage(string message) @@ -249,7 +249,26 @@ private void InstanceEncodeCompleted(object sender, EncodeCompletedEventArgs e) { this.IsEncoding = false; - this.ServiceLogMessage(e.Error != 0 ? string.Format("Encode Failed ({0})", e.Error) : "Encode Completed!"); + string completeMessage = "Job Completed!"; + switch (e.Error) + { + case 0: + break; + case 1: + completeMessage = "Job Cancelled!"; + break; + case 2: + completeMessage = string.Format("Job Failed. Check log and input settings ({0})", e.Error); + break; + case 3: + completeMessage = string.Format("Job Failed to Initialise. Check log and input settings ({0})", e.Error); + break; + default: + completeMessage = string.Format("Job Failed ({0})", e.Error); + break; + } + + this.ServiceLogMessage(completeMessage); // Handling Log Data string hbLog = this.ProcessLogs(this.currentTask.Destination, this.isPreviewInstance, this.currentConfiguration); diff --git a/win/CS/HandBrakeWPF/Services/UserSettingService.cs b/win/CS/HandBrakeWPF/Services/UserSettingService.cs index 4c181541a983..1570e6f8535f 100644 --- a/win/CS/HandBrakeWPF/Services/UserSettingService.cs +++ b/win/CS/HandBrakeWPF/Services/UserSettingService.cs @@ -304,7 +304,7 @@ private Dictionary GetDefaults() defaults.Add(UserSettingConstants.DefaultPlayer, false); // Experimental - defaults.Add(UserSettingConstants.RemoteServiceEnabled, false); + defaults.Add(UserSettingConstants.RemoteServiceEnabled, true); defaults.Add(UserSettingConstants.RemoteServicePort, 8037); // Misc