Click here to Skip to main content
65,938 articles
CodeProject is changing. Read more.
Articles
(untagged)

Bat file compiler

0.00/5 (No votes)
13 Feb 2008 1  
An article deskribing how to convert a bat file to an executable.

Contents

Introduction

This article shows how to compile a bat file into an executable file. The compiled executable can run without showing a window, and you can pass command line parameters to it as you would pass them to the bat file.

How the program works

The way this program compiles a bat file is quite tricky and weird, so I call this program a "Mock compiler". The bat file is not even parsed. This program creates another program, and adds the specified bat file as an embedded file in that program. When the generated program is executed, it extracts the embedded bat file to a temp folder, and runs it using the Process class. Simple, isn't it?

Using the code

In order to create another program from your application, you will need to create an instance of the CSharpCodeProvider class. The code snippet below shows how to do it:

using (CSharpCodeProvider code=new CSharpCodeProvider())
{
    CompilerParameters compar = new CompilerParameters();

    string option = "/target:winexe";

    // Set the icon for executable
    if (Properties.Settings.Default.Customicon && 
        File.Exists(Properties.Settings.Default.iconpath))
    {
       option += " " + "/win32icon:" + "\"" + 
                 Properties.Settings.Default.iconpath + "\"";
    }

    compar.CompilerOptions = option;
    compar.GenerateExecutable = true;
    compar.IncludeDebugInformation = false;

    //Add the bat file as an embedded resource
    if (File.Exists(filepath))
    {
       compar.EmbeddedResources.Add(filepath); 
    }

    compar.OutputAssembly = path;
    compar.GenerateInMemory = false; 

    //Add references
    compar.ReferencedAssemblies.Add("System.dll");
    compar.ReferencedAssemblies.Add("System.Data.dll");
    compar.ReferencedAssemblies.Add("System.Deployment.dll");
    compar.ReferencedAssemblies.Add("System.Drawing.dll");
    compar.ReferencedAssemblies.Add("System.Windows.Forms.dll");
    compar.ReferencedAssemblies.Add("System.Xml.dll");

    compar.TreatWarningsAsErrors = false;

    //Compile it
    //The code is included in the executable as a resource
    CompilerResults res = 
              code.CompileAssemblyFromSource(compar, Properties.Resources.Program);

    if (res.Errors.Count > 0)
    {
       result = false;
    }
    else
       result = true;
    }

When you run the generated executable, it will extract the bat file, process the command line arguments if any, and run it. If specified, the bat file will run without creating any window. Here is the code that shows how this is accomplished.

//This code requires System.Reflection namespace
//Extracts the bat file
private void extract()
{
    string name = Assembly.GetExecutingAssembly().GetManifestResourceNames()[0];
    hide = name.EndsWith("hideit.bat");

    Stream theResource = Assembly.GetExecutingAssembly().GetManifestResourceStream(name);

    BinaryReader br = new BinaryReader(theResource);
    FileStream fs = new FileStream(Environment.GetEnvironmentVariable("TEMP") + 
                    "\\it.bat", FileMode.Create);

    byte[] bt = new byte[theResource.Length];
    theResource.Read(bt, 0, bt.Length);
    fs.Write(bt, 0, bt.Length);

    br.Close();
    fs.Close();
}

//Process command line arguments
private string buildargument(string[] args)
{
    StringBuilder arg = new StringBuilder();

    for (int i = 0; i < args.Length; i++)
    {
       arg.Append(args[i] + " ");
    }

    return arg.ToString();
}

//Start the process
private void start(string[] args)
{
    ProcessStartInfo info = new ProcessStartInfo();
    info.FileName = Environment.GetEnvironmentVariable("TEMP") + "\\it.bat";

    //Specify argument
    info.Arguments = buildargument(args);

    //Hide the window if specified
    info.CreateNoWindow = hide;
    if (hide)
    {
        info.WindowStyle = ProcessWindowStyle.Hidden;
    }

    //Set the working directory for the bat file. This is important as the
    //bat file might use relative path
    info.WorkingDirectory = Application.StartupPath;

    //At last start the process
    Process proc = new Process();
    proc.StartInfo = info;
    proc.Start();
}

Points of interest

The way this program compiles bat files is quite tricky and a little bit strange.

History

  • 17 June, 2007 - Initial release.

License

This article has no explicit license attached to it but may contain usage terms in the article text or the download files themselves. If in doubt please contact the author via the discussion board below.

A list of licenses authors might use can be found here