views:

214

answers:

2

This is a bit involved, but please help. I'm starting to fall way behind due to this problem. God bless you for your attention to this.

I've been using C# System.Diagnostics.Process to monitor the output of a command line utility.

The process I'm monitoring "internally" launches a second process, and as soon as it does, I receive no further output from the process object.

What's frustrating, is, if you execute the very same command (that I'm launching with the System.Diagnostics.Process object) with cmd.exe (manually), the console outputs every line I need to be seeing in my c# app! However, if I (for testing purposes) launch cmd.exe with the System.Diagnostics.Process object, and run the command, it still stops outputting at the same point that it did previously (launching process1.exe directly); at the point the second.exe is utilised. I thought this test would consolidate all output from all processes involved, but it didn't. How can I get all this output into my C# application?

Conceptually, this is my problem (I think). If you need more details, please go here: http://www.lonniebest.com/Help/

+1  A: 

Do you need to be launching cmd.exe at all? Can't you just initiate the process for rsync directly in your Process and then use something like the techniques described in this question to catch the output from the command execution so you can work with them in your code?

glenatron
No, I only tried launching cmd.exe, instead, because I was hoping it would show all output from all processes. I initially tried it the way you are mentioning, and the output (using the techniques in the question you referenced) ceased after the first process started to utilise the second process that the first process calls internally.
LonnieBest
+5  A: 

The reason for this is that the System.Diagnostics.Process is literally only monitoring the process to which it is hooked.

One way to circumvent this problem would be for your first application to output when it is starting the second application, and when that output is received, monitor from your main application for the creation of the process from the (now third) application. Once the third application is started, it should appear in the System.Diagnostics.Process.GetProcesses() array, and you can then attach to it's OutputDataReceived event.

Your code would then look something like this (untested):

private void firstProcess_OutputDataReceived(object sender, System.Diagnostics.DataReceivedEventArgs e)
{
    if (e.Data == "Starting next process")
    {
        System.Diagnostics.Process newProcess = null;

        while (newProcess == null)
        {
            System.Diagnostics.Process[] procs = System.Diagnostics.Process.GetProcesses();

            foreach (System.Diagnostics.Process proc in procs)
            {
                if (proc.ProcessName == "newProcess")
                {
                    newProcess = proc;
                    break;
                }
            }

            System.Threading.Thread.Sleep(100);
        }

        newProcess.OutputDataReceived += new System.Diagnostics.DataReceivedEventHandler(newProcess_OutputDataReceived);
    }
}

void newProcess_OutputDataReceived(object sender, System.Diagnostics.DataReceivedEventArgs e)
{
    // Do something with your data received here.
}

Note that this is just a sample, and if your third process fails to start, or ends too quickly, then this method will hang in an infinite loop. This sample is simply meant to provide you with the knowledge to build something that works in your specific case, which I'm not totally familiar with. You should at a very minimum make sure that the while loop will not continue forever, and you'll probably want to make a few other adjustments as well.

EDIT: Alternately, if you can't modify the source to the first application, you could simply create a new thread that monitored in this manner (using the while loop) constantly and handled the output from the third process in a separate class, or simply re-routed the output from the third process into the handler for the output of the second process so that you could have a single method handling all of the output for both processes.

md5sum
First of, I'm amazed at how quickly you've responded with such detail and I really appreciate your help. From the output of the first process, I'm able to determine the point at which the 2nd process "was" launched.
LonnieBest
To bad there isn't a object in the .Net framework that is as good of a "process output consolidator" as cmd.exe is (when run manually).
LonnieBest
Glad I could help :D
md5sum
Thanks for your efforts, but I still ran into this:http://stackoverflow.com/questions/2095826/c-redirect-standard-output-of-a-process-that-is-already-running
LonnieBest