What is the difference between Process.Start() and running manually?

3k Views Asked by At

I am creating an app for the company I work for which will backup the data for some software we supply. It has options to add SQL databases, folders, files and reg keys to the backup job which are then put into a Zip file. There are also options to restore the backup when required.

I am having a problem with the registry backup and restore. I have been through many iterations of trying to get this to work but I have simplified it in a way that best illustrates my problem that I hope someone can help with.

Essentially I am importing a ".reg" file by using the regedit.exe with command line arguments. I have tried doing this by building it with ProcessStartInfo() but it did not work. So to test the problem I am creating a batch file instead and running it as follows:

    File.WriteAllText("ImportReg.bat", "regedit /s /i MyRegFile.reg");
    Process.Start("ImportReg.bat");

This however does not work.

The batch file is created successfully and the REG file is valid. Both files are in the same location as the EXE to so I don't think the issue should be with which directory is being used. If I run the batch file myself from this location it successfully imports the registry file. I am a full admin on the machine I am testing this on.

I have also had issues with exporting from the registry. Some parts of the registry export fine and others don't. However, if I set the export up as a batch file like the import above then it works every time.

Can anyone help with this? I cannot see why my batch file works, but not when it's run via Process.Start. Any suggestions would be massively appreciated.

3

There are 3 best solutions below

1
On

Could it be a permissions issue?

Regedit is automatically elevated when you double-click on it. But when running through your program, it is not. Perhaps that's why it lets you export some parts of the regiistry, but not others.

0
On

You probably passed "regedit /s /i MyRegFile.reg" as the executable name. Pass "regedit.exe" and arguments "/s /i MyRegFile.reg".

22
On

When a program is run even if the logged on user is an administrator it is not run with elevated permissions. Option A) Change the Process.Start to use 'runas' to elevate the created process

...
using System.Diagnostics;
...

System.Diagnostics.Process proc = new System.Diagnostics.Process();
  proc.StartInfo.FileName        = "C:\\Windows\\system32\\notepad.exe";
  proc.StartInfo.Verb            = "runas"; // Elevate the application
  proc.StartInfo.UseShellExecute = true;
proc.Start();

Option B) To elevate the permissions by adding a manifest file

http://msdn.microsoft.com/en-us/library/bb384691.aspx

To set this linker option in the Visual Studio development environment

  1. Open the project's Property Pages dialog box. For details, see How to: Open Project Property Pages.
  2. Expand the Configuration Properties node.
  3. Expand the Linker node.
  4. Select the Manifest File property page.
  5. Modify the Enable User Account Control (UAC), UAC Execution Level, and UAC Bypass UI Protection properties.

The manifest file will look something like this

<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<assembly xmlns="urn:schemas-microsoft-com:asm.v1" manifestVersion="1.0">
    <v3:trustInfo xmlns:v3="urn:schemas-microsoft-com:asm.v3">
        <v3:security>
            <v3:requestedPrivileges>
                <v3:requestedExecutionLevel level="requireAdministrator"/>
            </v3:requestedPrivileges>
        </v3:security>
    </v3:trustInfo>
</assembly>

Option C), add the following code (which uses the verb runas, to respawn your program with it's arguments to elevate a duplicate child process) Windows 7 and Vista UAC - Programmatically requesting elevation in C#