I cannot get a Terminal window to open from Process.Start().
The following should execute a shell script in a new Terminal window.
using(var process=new Process())
process.StartInfo.FileName=“/bin/sh”;
process.StartInfo.Arguments="shell.sh;
process.StartInfo.WorkingDirectory=“path/to/working/directory”;
process.StartInfo.CreateNoWindow=false;
process.StartInfo.UseShellExecute=false;
process.StartInfo.RedirectStandardOutput=false;
process.Start();
process.WaitForExit();
process.Close();
This SHOULD open Terminal and run shell.sh.
Instead, it runs shell.sh silently without opening Terminal.
Yes, I could reroute the output from shell.sh back into Unity. But if I’m running something like npm install or pod install, or something like fastlane sigh where I need to enter app store credentials, running silently isn’t an option.
The ONLY way I can get Terminal to open is by using
process.StartInfo.FileName=“open”;
and ensuring that my .sh files are set to open in Terminal by default.
2 downsides to this method:
This instance ignores the WorkingDirectory argument and defaults to my home directory.
I cannot pass parameters into the shell script by appending onto the end of the Arguments string.
And as a result, I can’t pass in my working directory in order to cd to the appropriate location from within the shell script.
What am I missing here? Why isn’t Terminal opening when I using FileName=“/bin/sh”, and how should I pass params using FileName=“open”?
If you see someone who needs to know about code tags, please link them to this post: Please use code tags when posting code. You can "tag" your code by typing around your code. There is no overt "Code" tag button that automatically tags a...
Reading time: 9 mins 🕑
Likes: 83 ❤
This SHOULD open Terminal and run shell.sh.
Instead, it runs shell.sh silently without opening Terminal.
No it shouldn’t… it should open /bin/sh and run shell.sh
/bin/sh is not a terminal, it’s the shell, which is by design headless. You must capture input and output in linux for the shell… this is what programs like ‘xterm’ or ‘terminal’ or the sort do. They give a gui to the shell display output, and directing input.
The ONLY way I can get Terminal to open is by using
process.StartInfo.FileName=“open”;
and ensuring that my .sh files are set to open in Terminal by default.
Yep, because ‘open’ tells your OS to open the file with the program it thinks it aught to. So in your desktop environment it’s probably defaulted to use something like ‘xterm’ or another terminal emulator and then runs the script.
Thanks for your reply
@lordofduct
. I was under the impression that running /bin/sh (or /bin/bash) out of Unity with certain params set on the Process.StartInfo object would open the shell in Terminal. But as you’ve explained, that is not the case.
Do you know of a way to pass arguments to a file when using open? Appending --args to the Arguments string passes those arguments to the default application (in this case Terminal) and not to the shell script.
well, for one, don’t use ‘open’. That just asks the OS to figure out what should be used to open some given file based. In your case a *.sh file, which the OS says to use your terminal.
Instead, open the terminal, and pass arguments to that. You can tell a terminal to execute something with the -e command:
In my case, I use Linux Mint, so I used ‘mate-terminal’:
using (var proc = new Process ()) {
proc.StartInfo.FileName = "mate-terminal";
proc.StartInfo.Arguments = "-e ./test.sh";
proc.StartInfo.WorkingDirectory = "/home/dylan/Temp";
proc.StartInfo.CreateNoWindow = false;
proc.StartInfo.UseShellExecute = false;
proc.StartInfo.RedirectStandardOutput = false;
proc.Start ();
proc.WaitForExit ();
proc.Close ();
Thing is, this requires you to know the terminal application used.
Another option is to use the $TERM environment variable, as long as it is set correctly. (go into a terminal, type ‘echo $TERM’, if the output is that of the terminal you want to use and is installed… it should work)
var term = System.Environment.GetEnvironmentVariable("TERM");
using (var proc = new Process ()) {
proc.StartInfo.FileName = term;
proc.StartInfo.Arguments = "-e ./test.sh";
proc.StartInfo.WorkingDirectory = "/home/dylan/Temp";
proc.StartInfo.CreateNoWindow = false;
proc.StartInfo.UseShellExecute = false;
proc.StartInfo.RedirectStandardOutput = false;
proc.Start ();
proc.WaitForExit ();
proc.Close ();
Unfortunately, when using the MonoDevelop debugger it doesn’t give you access to the environment variables (because, of course it doesn’t…). Not sure if Unity in debug mode with have access (I haven’t used the linux build of the editor). It ‘should’ work if built though… don’t currently have the time to confirm though. Give it a try.
Of course, another issue is that when doing this the terminal actually spins up its own process when executing that script. So this process won’t wait for it. You need to for it to… so add the --disable-factory command as well.
using (var proc = new Process ()) {
proc.StartInfo.FileName = "mate-terminal";
proc.StartInfo.Arguments = "--disable-factory -e ./test.sh";
proc.StartInfo.WorkingDirectory = "/home/dylan/Temp";
proc.StartInfo.CreateNoWindow = false;
proc.StartInfo.UseShellExecute = false;
proc.StartInfo.RedirectStandardOutput = false;
proc.Start ();
proc.WaitForExit ();
proc.Close ();
Is there a reason you need to show a terminal window when you do this?
Is this just so you can display the output from the shell script?
If so… you can capture the output from shell and display that.
using (var proc = new Process ()) {
proc.StartInfo.FileName = "/bin/sh";
proc.StartInfo.Arguments = "test.sh";
proc.StartInfo.WorkingDirectory = "/home/dylan/Temp";
proc.StartInfo.CreateNoWindow = false;
proc.StartInfo.UseShellExecute = false;
proc.StartInfo.RedirectStandardOutput = false;
proc.Start ();
proc.WaitForExit ();
string result = proc.StandardOutput.ReadToEnd ();
Debug.Log(result);
proc.Close ();
I landed on this:
process.StartInfo.FileName = "osascript";
process.StartInfo.Arguments = string.Format("-e 'tell application \"Terminal\" to do script \"cd {0} && ./{1} \"'", scriptPath, script);
process.StartInfo.WorkingDirectory = workingDirectory;
process.StartInfo.UseShellExecute = false;
process.Start();
process.WaitForExit();
process.Close();