-
Notifications
You must be signed in to change notification settings - Fork 636
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Cherry-pick PRs 14703, 14637 (#15020)
* DYN-6527: Fix graph update for primitive input nodes that are first initialized to null (#14703) * remove coreclr-ncalc references * add failing test for dropdown node * cleanup * update tests * attempt initial fix * cleanup * update test * review comments * add code comments * cherry-pick PR 14637 --------- Co-authored-by: Michael Kirschner <mjk.kirschner@gmail.com>
- Loading branch information
1 parent
cdf8c5a
commit 0dc249d
Showing
11 changed files
with
434 additions
and
234 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,175 @@ | ||
using System; | ||
using System.ComponentModel; | ||
using System.Diagnostics; | ||
using System.IO; | ||
using System.Reflection; | ||
using System.Threading.Tasks; | ||
|
||
namespace Dynamo.Utilities | ||
{ | ||
/// <summary> | ||
/// Base class for Dynamo CLI wrappers | ||
/// </summary> | ||
internal abstract class CLIWrapper : IDisposable | ||
{ | ||
protected const string endOfDataToken = @"<<<<<Eod>>>>>"; | ||
protected const string startofDataToken = @"<<<<<Sod>>>>>"; | ||
protected readonly Process process = new Process(); | ||
protected bool started; | ||
internal event Action<string> MessageLogged; | ||
|
||
public virtual void Dispose() | ||
{ | ||
process.ErrorDataReceived -= Process_ErrorDataReceived; | ||
KillProcess(); | ||
} | ||
|
||
/// <summary> | ||
/// Start the process. | ||
/// </summary> | ||
/// <param name="relativeEXEPath">relative path to the exe to start.</param> | ||
/// <param name="argString">argument string to pass to process.</param> | ||
protected virtual void StartProcess(string relativeEXEPath, string argString) | ||
{ | ||
ProcessStartInfo startInfo = new ProcessStartInfo | ||
{ | ||
CreateNoWindow = true, | ||
RedirectStandardOutput = true, | ||
RedirectStandardInput = true, | ||
RedirectStandardError = true, | ||
|
||
UseShellExecute = false, | ||
Arguments = argString, | ||
FileName = GetToolPath(relativeEXEPath) | ||
}; | ||
|
||
process.StartInfo = startInfo; | ||
try | ||
{ | ||
process.Start(); | ||
started = true; | ||
//the only purpose here is to avoid deadlocks when std err gets filled up 4kb | ||
//in long running processes. | ||
process.ErrorDataReceived += Process_ErrorDataReceived; | ||
process.BeginErrorReadLine(); | ||
|
||
} | ||
catch (Win32Exception) | ||
{ | ||
// Do nothing | ||
} | ||
} | ||
|
||
private void Process_ErrorDataReceived(object sender, DataReceivedEventArgs e) | ||
{ | ||
//do nothing, we just want to empty the error stream. | ||
} | ||
|
||
|
||
|
||
/// <summary> | ||
/// Kill the CLI tool - if running | ||
/// </summary> | ||
protected void KillProcess() | ||
{ | ||
if (started) | ||
{ | ||
if (!process.HasExited) | ||
{ | ||
process.Kill(); | ||
} | ||
started = false; | ||
} | ||
process.Dispose(); | ||
} | ||
/// <summary> | ||
/// Compute the location of the CLI tool. | ||
/// </summary> | ||
/// <returns>Returns full path to the CLI tool</returns> | ||
protected static string GetToolPath(string relativePath) | ||
{ | ||
var rootPath = Path.GetDirectoryName(Assembly.GetExecutingAssembly().Location) ?? throw new ArgumentNullException(nameof(Path.GetDirectoryName)); | ||
var toolPath = Path.Combine(rootPath,relativePath); | ||
return toolPath; | ||
} | ||
/// <summary> | ||
/// Read data from CLI tool | ||
/// </summary> | ||
/// <param name="timeoutms">will return empty string if we don't finish reading all data in the timeout provided in milliseconds.</param> | ||
/// <returns></returns> | ||
protected virtual async Task<string> GetData(int timeoutms) | ||
{ | ||
var readStdOutTask = Task.Run(() => | ||
{ | ||
if (process.HasExited) | ||
{ | ||
return string.Empty; | ||
} | ||
|
||
using (var writer = new StringWriter()) | ||
{ | ||
var done = false; | ||
var start = false; | ||
while (!done) | ||
{ | ||
try | ||
{ | ||
var line = process.StandardOutput.ReadLine(); | ||
MessageLogged?.Invoke(line); | ||
if (line == null || line == startofDataToken) | ||
{ | ||
start = true; | ||
continue; //don't record start token to stream. | ||
} | ||
|
||
if (line == null || line == endOfDataToken) | ||
{ | ||
done = true; | ||
} | ||
else | ||
{ | ||
//if we have started recieving valid data, start recording | ||
if (!string.IsNullOrWhiteSpace(line) && start) | ||
{ | ||
writer.WriteLine(line); | ||
} | ||
} | ||
} | ||
catch (Exception) | ||
{ | ||
KillProcess(); | ||
return GetCantCommunicateErrorMessage(); | ||
} | ||
} | ||
|
||
return writer.ToString(); | ||
} | ||
}); | ||
var completedTask = await Task.WhenAny(readStdOutTask, Task.Delay(TimeSpan.FromMilliseconds(timeoutms))); | ||
//if the completed task was our read std out task, then return the data | ||
//else we timed out, so return an empty string. | ||
return completedTask == readStdOutTask ? readStdOutTask.Result : string.Empty; | ||
} | ||
|
||
protected void RaiseMessageLogged(string message) | ||
{ | ||
MessageLogged?.Invoke(message); | ||
} | ||
|
||
/// <summary> | ||
/// Can't start Error message | ||
/// </summary> | ||
/// <returns>Returns error message</returns> | ||
protected abstract string GetCantStartErrorMessage(); | ||
|
||
|
||
/// <summary> | ||
/// Can't communicate Error message | ||
/// </summary> | ||
/// <returns>Returns error message</returns> | ||
protected abstract string GetCantCommunicateErrorMessage(); | ||
|
||
|
||
|
||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.