Wednesday, August 27, 2008
System.Environment.Exit and Mono.GetOptions
C# is apparently such a well-structured language (or its authors wished it to be one) so that it doesn't have normal "exit" function. Indeed, why does one need a special "exit" utility? Program should end simply when "Main" function reaches its end. Simple, ah?
Well, let's take a step back. Microsoft .NET platform apparently does not have its own "standard" argument-parsing utility similar to GNU "getopt". Obviously, it isn't a huge problem to do a straightforward port of "getopt" to C#; see one such port (not free, unfortunately) here. However, Mono project provides a better alternative: Mono.GetOptions assembly.
Blog post from PumaCode.org blog gives a very neat overview of how it is done; basically, for testing you can create one C# file like that:
using System; using System.Reflection; using Mono.GetOptions; // Attributes visible in " --help" [assembly: AssemblyTitle ("go.exe")] [assembly: AssemblyVersion ("1.0.*")] [assembly: AssemblyDescription ("Mono.GetOptions Sample Program")] [assembly: AssemblyCopyright ("Public Domain")] // This is text that goes after " [options]" in help output. [assembly: Mono.UsageComplement ("")] // Attributes visible in " -V" [assembly: Mono.About("Insert About Text Here.")] [assembly: Mono.Author ("Your name here")] class SampleOptions : Options { // Long option is the variable name ("--file"), short option is -f [Option ("Write report to FILE", 'f')] public string file; // Long option is the variable name ("--quiet"), short option is -q [Option ("don't print status messages to stdout", 'q')] public bool quiet; // Long option is as specified ("--use-int"), no short option [Option ("Sample int option", "use-int")] public int use_int; public SampleOptions () { base.ParsingMode = OptionsParsingMode.Both; } } class TestApp { public static void Main (string[] args) { SampleOptions options = new SampleOptions (); options.ProcessArgs (args); Console.WriteLine ("Specified Program Options:"); Console.WriteLine ("\t file: {0}", options.file); Console.WriteLine ("\t quiet: {0}", options.quiet); Console.WriteLine ("\t use_int: {0}", options.use_int); Console.WriteLine ("Remaining Program Options:"); foreach (string s in options.RemainingArguments) { Console.WriteLine ("\t{0}", s); } } }
and try to compile and run it. With mono, it is as simple as:
gmcs -r:Mono.GetOptions getopt.cs
Under .NET, we have to copy Mono.GetOptions.dll from somewhere first. If you have Mono installed on some Linux machine, you can find this DLL under "/usr/lib/mono/gac".
Of course, you need to know where to copy it it to and how to use it. This question has two parts: (a) how to compile code which uses Mono.GetOptions; (b) How to run/distribute code which uses Mono.GetOptions .
How to compile.
You can use compile-option "-r:<full path to Mono.GetOptions.dll>". This will work.
How to make it work by simply using -r:Mono.GetOptions.dll with no path? That's a good question which I cannot answer at this point. A "standard" location for "global" assemblies is "c:\WINNT\assembly" (or similar); note that you can only copy something there with Windows Explorer UI, since it is using special Shell Extension to properly distribute DLLs to some internal directories, not visible from the shell. If you've copied your DLL there, you can reference it as:
/r:c:\WINNT\assembly\GAC_MSIL\Mono.GetOptions\2.0.0.0__0738eb9f132ed756\Mono.GetOptions.dll
Or, alternatively, you can append the above line to your CSC.RSP file (e.g. "c:\WINNT\Microsoft.NET\Framework\v3.5\csc.rsp")
How to run.
This is simple enough - if you have copied your assembly to "c:\WINNT\assembly", it will run fine.
End of story?
No, not yet.
Your test executable will indeed run... till you'll try to copy it to network share and run from there. If you do that and execute "getopt.exe -help", you'll observe "security violation" failure.
One might ask, what kind of security could be violated by parsing a command line?
It is very simple, indeed. As one can see from the source code, function OptionList.ProcessArgs() calls function System.Environment.Exit(), and as it follows from both mono documentation and Microsoft documentation, this function for some reason needs permission to "execute unmanaged code" to run; by default, such permission is not afforded to programs run from "local intranet", including mounted shares.
Is there any way to override this default and grant all required permissions? Perhaps. There is (or was?) utility called caspol.exe which could help you; whether this utility is still supported is not clear. Rather, Microsoft suggest using UI-based configuration tool (part of Windows "Administrative tools", available from Control Panel). When I tried changing that saying that policies from "My computer" shall apply to everything coming from "Local intranet", it indeed fixed the problem... except that as a result Windows Live Writer refused to start! Fortunately, this "configuration tool" has an option to restore default behavior....
.... And all this trouble from simple desire to use some kind of argument-parsing in C#.....
A much better design, and only requires you to copy a single file into your project.
Check it out here:
http://www.ndesk.org/Options
<< Home