【问题标题】:Run new process as different user in vb.net在 vb.net 中以不同用户身份运行新进程
【发布时间】:2010-10-03 13:57:14
【问题描述】:

我目前正在使用本土方法在 Vista 中以不同用户的身份运行进程,我无法摆脱这种感觉是 hack-ish 并且不太理想(除了它会破坏 UAC 的事实,使我的应用程序因安全异常而崩溃,并迫使我完全禁用 UAC)。我的流程由两个项目(所以两个 EXE 文件)组成——一个“接口”和一个“启动存根”——流程如下:

  1. 用户有一个启动“Interface.exe notepad.exe”的快捷方式
  2. Interface.exe 有一个表单,询问他们想要使用的凭据
  3. Interace.exe 使用 ProcessStartInfo 作为新用户创建 LaunchStub.exe (LS) 的实例
  4. LS 使用 ProcessStartInfo(ShellExecute 设置为 true)来启动请求的文件,并且由于它已经以请求的用户身份运行,因此新进程也是如此。

我有一个两步过程的原因是我希望用户能够右键单击操作系统对(.EXE、.SQL、.MSC 等)具有默认操作的任何文件并启动它,而 ProcessStartInfo 只支持启用“UseShellExecute”,但该开关阻止我使用新凭据,所以我一次只能做一个。

这会导致一些问题 - 首先,用户必须已经存在于计算机上,这意味着他们之前必须在本地登录。如果该用户没有本地配置文件,则有时会启动所请求的应用程序,但我会收到注册表和配置文件异常,因为该应用程序希望存在尚不存在的事物(例如注册表中的 HKCU 配置单元,而用户不存在)因为他们从未登录过)。

我知道我应该能够将我的应用程序的权限“提升”给他们请求的用户,启动我的新进程,然后撤消提升,但我无法找到一个好的代码示例那,我不确定它是否允许以完全不同的用户身份运行。这一切有意义吗?我不禁觉得有更好的方法来做到这一点。


更新:我刚刚尝试了some Impersonation code,我在网上找到了,但无济于事。与 ProcessStartInfo 结合使用时,它似乎仍然使用我当前的登录名来启动进程,而不是我提供的登录名,即使我已经使用提供的凭据激活了模拟。

【问题讨论】:

    标签: vb.net impersonation runas security-context startprocessinfo


    【解决方案1】:

    您可以尝试从您的应用程序运行 runas。一些示例和选项here

    【讨论】:

      【解决方案2】:

      您可能必须使用 Win32 API 创建自己的“shell”函数。

      使用 CreateProcessWithLogonW API,您可以在不同的凭据下创建新进程,并可选择加载用户配置文件信息。

      如果你替换在下面的代码sn-p中

      • 用户名 - 使用您的用户名
      • 域 - 使用您的域或“vbNullString”
      • 密码 - 使用您的密码
      • 参数 4 - 将 0 替换为 'LOGON WITH PROFILE' 以加载指定的用户配置文件。

      有关详细信息,请参阅CreateProcessWithLogonW API 的文档。走这条路,您可以完全控制并完全负责启动应用程序。

      同样,这只是一个示例,您可能需要稍微尝试一下才能让它做您想做的事情。

      
      Imports System.Runtime.InteropServices
      
      Public Module modShell
      
          <StructLayout(LayoutKind.Sequential)> _
          Public Structure STARTUPINFO
              Public cb As Integer
              Public lpReserved As String
              Public lpDesktop As String
              Public lpTitle As String
              Public dwX As Integer
              Public dwY As Integer
              Public dwXSize As Integer
              Public dwYSize As Integer
              Public dwXCountChars As Integer
              Public dwYCountChars As Integer
              Public dwFillAttribute As Integer
              Public dwFlags As Integer
              Public wShowWindow As Short
              Public cbReserved2 As Short
              Public lpReserved2 As Integer
              Public hStdInput As Integer
              Public hStdOutput As Integer
              Public hStdError As Integer
          End Structure
      
          <StructLayout(LayoutKind.Sequential)> _
          Public Structure PROCESS_INFORMATION
              Public hProcess As IntPtr
              Public hThread As IntPtr
              Public dwProcessId As Integer
              Public dwThreadId As Integer
          End Structure
      
          Public Declare Unicode Function CreateProcessWithLogonW Lib "Advapi32" (ByVal lpUsername As String, ByVal lpDomain As String, ByVal lpPassword As String, ByVal dwLogonFlags As Int32, ByVal lpApplicationName As String, ByVal lpCommandLine As String, ByVal dwCreationFlags As Int32, ByVal lpEnvironment As IntPtr, ByVal lpCurrentDirectory As String, ByRef si As STARTUPINFO, ByRef pi As PROCESS_INFORMATION) As Integer
          Public Declare Function CloseHandle Lib "kernel32" (ByVal hObject As IntPtr) As Integer
      
          Public Const LOGON_WITH_PROFILE As Int32 = &H1
      
          Public Const NORMAL_PRIORITY_CLASS As Int32 = &H20&
      
          Public Const STARTF_USESHOWWINDOW As Int32 = &H1
          Public Const SW_HIDE As Int16 = 0
          Public Const SW_SHOW As Int16 = 5
      
          Public Function Shell(ByVal strCmdLine As String, ByVal strCurrentDirectory As String) As Boolean
      
              Dim pi As PROCESS_INFORMATION
              Dim si As New STARTUPINFO
      
              si.cb = Marshal.SizeOf(si)
              si.dwFlags = STARTF_USESHOWWINDOW
              si.wShowWindow = SW_SHOW
      
              Dim result As Integer = CreateProcessWithLogonW("username", "domain", "password", 0, vbNullString, strCmdLine, NORMAL_PRIORITY_CLASS, IntPtr.Zero, strCurrentDirectory, si, pi)
      
              If result <> 0 Then
                  Call CloseHandle(pi.hThread)
                  Call CloseHandle(pi.hProcess)
              Else
                  Return False
              End If
      
              Return True
      
          End Function
      
      End Module
      
      

      【讨论】:

      • 我会试一试,看看会发生什么。如果我可以在不必使用第二个 EXE 作为存根的情况下做到这一点,我会很高兴,如果在启用 UAC 时它没有崩溃(我会处理提示),那就更高兴了。试了之后我就bakc了。
      【解决方案3】:

      试试这个模块:

      Module Impersonation
      
      #Region "API Structures"
          <StructLayout(LayoutKind.Sequential)> _
            Public Structure PROCESS_INFORMATION
              Dim hProcess As System.IntPtr
              Dim hThread As System.IntPtr
              Dim dwProcessId As Integer
              Dim dwThreadId As Integer
          End Structure
      
          <StructLayout(LayoutKind.Sequential)> _
           Public Structure STARTUPINFO
              Dim cb As Integer
              Dim lpReserved As System.IntPtr
              Dim lpDesktop As System.IntPtr
              Dim lpTitle As System.IntPtr
              Dim dwX As Integer
              Dim dwY As Integer
              Dim dwXSize As Integer
              Dim dwYSize As Integer
              Dim dwXCountChars As Integer
              Dim dwYCountChars As Integer
              Dim dwFillAttribute As Integer
              Dim dwFlags As Integer
              Dim wShowWindow As Short
              Dim cbReserved2 As Short
              Dim lpReserved2 As System.IntPtr
              Dim hStdInput As System.IntPtr
              Dim hStdOutput As System.IntPtr
              Dim hStdError As System.IntPtr
          End Structure
      #End Region
      
      #Region "API Constants"
          Private Const LOGON_NETCREDENTIALS_ONLY As Integer = &H2
          Private Const NORMAL_PRIORITY_CLASS As Integer = &H20
          Private Const CREATE_DEFAULT_ERROR_MODE As Integer = &H4000000
          Private Const CREATE_NEW_CONSOLE As Integer = &H10
          Private Const CREATE_NEW_PROCESS_GROUP As Integer = &H200
          Private Const LOGON_WITH_PROFILE As Integer = &H1
      #End Region
      
      #Region "API Functions"
          Private Declare Unicode Function CreateProcessWithLogon Lib "Advapi32" Alias "CreateProcessWithLogonW" _
              (ByVal lpUsername As String, _
               ByVal lpDomain As String, _
               ByVal lpPassword As String, _
               ByVal dwLogonFlags As Integer, _
               ByVal lpApplicationName As String, _
               ByVal lpCommandLine As String, _
               ByVal dwCreationFlags As Integer, _
               ByVal lpEnvironment As System.IntPtr, _
               ByVal lpCurrentDirectory As System.IntPtr, _
               ByRef lpStartupInfo As STARTUPINFO, _
               ByRef lpProcessInfo As PROCESS_INFORMATION) As Integer
      
          Private Declare Function CloseHandle Lib "kernel32" (ByVal hObject As System.IntPtr) As Integer
      
      #End Region
      
          Public Sub RunProgram(ByVal UserName As String, ByVal Password As String, ByVal Domain As String, ByVal Application As String, ByVal CommandLine As String)
      
              Dim siStartup As STARTUPINFO
              Dim piProcess As PROCESS_INFORMATION
              Dim intReturn As Integer
      
              If CommandLine Is Nothing OrElse CommandLine = "" Then CommandLine = String.Empty
      
              siStartup.cb = Marshal.SizeOf(siStartup)
              siStartup.dwFlags = 0
      
              intReturn = CreateProcessWithLogon(UserName, Domain, Password, LOGON_WITH_PROFILE, Application, CommandLine, _
              NORMAL_PRIORITY_CLASS Or CREATE_DEFAULT_ERROR_MODE Or CREATE_NEW_CONSOLE Or CREATE_NEW_PROCESS_GROUP, _
              IntPtr.Zero, IntPtr.Zero, siStartup, piProcess)
      
              If intReturn = 0 Then
                  Throw New System.ComponentModel.Win32Exception(Marshal.GetLastWin32Error())
              End If
      
              CloseHandle(piProcess.hProcess)
              CloseHandle(piProcess.hThread)
      
          End Sub
      
      End Module
      

      使用 Runprogram() 以 user/pw y 之类的方式启动您的程序。 Programm 仅表示 .exe,参数写入“命令​​行”

      【讨论】:

      • 这个模块与之前发布的模块有何不同?老实说,它看起来很相似,所以我只是想确保我没有遗漏一些明显的优势。我还没有接受答案,但之前的代码一直为我工作。
      • 是的,它非常相似,它只是第二个编写的函数,围绕 apifunktion CreateProcessWithLogonW API - 带有错误处理和成功反馈的 ready_to_use ...
      【解决方案4】:

      如果您想使用与当前运行进程不同的凭据启动应用程序,您可以使用 .Net Process 类。

      this.Process = new Process();
      
      this.Process.StartInfo.Arguments = "Arguments";
      this.Process.StartInfo.FileName = "C:\your.exe";
      this.Process.StartInfo.UserName = "UserName";
      string password = "some password";
      
      this.Process.StartInfo.Password.Clear();
      foreach (char c in password)
      {
          this.Process.StartInfo.Password.AppendChar(c);
      }
      
      
      //allow the process to raise events
      this.Process.EnableRaisingEvents = true;
      this.Process.StartInfo.ErrorDialog = false;
      //Method for handling the exit event
      this.Process.Exited += new EventHandler(ApplicationProcess_Exited);
      
      //Set the application directory as the current working directory
      Environment.CurrentDirectory = System.IO.Directory.GetParent("C:\").ToString();
      
      if (this.Process.Start())
      {
          // Do something on start
      }
      

      【讨论】:

        猜你喜欢
        • 1970-01-01
        • 2012-11-29
        • 1970-01-01
        • 2010-12-18
        • 1970-01-01
        • 1970-01-01
        • 2019-07-16
        • 1970-01-01
        • 1970-01-01
        相关资源
        最近更新 更多