通过控制台或Windows服务以编程方式创建Windows会话

时间:2018-10-14 09:03:21

标签: c# winforms windows-services activex rdp

如何以编程方式登录Windows以创建Windows登录会话? 我需要一种从WinForms应用程序,控制台应用程序以及(最重要的)Windows服务中运行的方法。

另一个要求是我需要它在运行程序/服务的本地系统以及远程系统上工作。

如果有办法使用pInvoke / Win32 API来做到这一点,我也很乐意。

我在研究中发现了这些类似的问题/答案:

Programmatically create and launch and RDP session (without gui)

这里的答案说可以,但是给出了链接,但是链接中的示例代码不起作用

Create a Windows Session from a service via the Win32 API

无法解决所提问题

Create Windows session programmatically

没有解决方案,但OP在评论中提到http://freerdp.com为他工作。

1 个答案:

答案 0 :(得分:0)

我创建了一个简单的实用程序,相信可以满足问题中的所有要求。您需要向Microsoft终端服务Active Client 1.0类型库(ActiveX)添加COM引用。

我认为它可能无法在本地计算机上创建会话,但是我在2012R2中作为服务运行进行了测试,实际上可以。可以从WinForms应用程序或控制台应用程序中调用相同的确切方法。从WinForms或控制台应用程序启动时,该表单会显示几秒钟,因此我确保将控件设置为enabled = false,以使其无法与之交互。

using System;
using System.Diagnostics;
using System.Threading;
using System.Threading.Tasks;
using System.Windows.Forms;
using AxMSTSCLib;

namespace Utility.RemoteDesktop
{
    public class Client
    {
        private int LogonErrorCode { get; set; }

        public void CreateRdpConnection(string server, string user, string domain, string password)
        {
            void ProcessTaskThread()
            {
                var form = new Form();
                form.Load += (sender, args) =>
                {
                    var rdpConnection = new AxMSTSCLib.AxMsRdpClient9NotSafeForScripting();
                    form.Controls.Add(rdpConnection);
                    rdpConnection.Server = server;
                    rdpConnection.Domain = domain;
                    rdpConnection.UserName = user;
                    rdpConnection.AdvancedSettings9.ClearTextPassword = password;
                    rdpConnection.AdvancedSettings9.EnableCredSspSupport = true;
                    if (true)
                    {
                        rdpConnection.OnDisconnected += RdpConnectionOnOnDisconnected;
                        rdpConnection.OnLoginComplete += RdpConnectionOnOnLoginComplete;
                        rdpConnection.OnLogonError += RdpConnectionOnOnLogonError;
                    }
                    rdpConnection.Connect();
                    rdpConnection.Enabled = false;
                    rdpConnection.Dock = DockStyle.Fill;
                    Application.Run(form);
                };
                form.Show();
            }

            var rdpClientThread = new Thread(ProcessTaskThread) { IsBackground = true };
            rdpClientThread.SetApartmentState(ApartmentState.STA);
            rdpClientThread.Start();
            while (rdpClientThread.IsAlive)
            {
                Task.Delay(500).GetAwaiter().GetResult();
            }
        }

        private void RdpConnectionOnOnLogonError(object sender, IMsTscAxEvents_OnLogonErrorEvent e)
        {
            LogonErrorCode = e.lError;
        }
        private void RdpConnectionOnOnLoginComplete(object sender, EventArgs e)
        {
            if (LogonErrorCode == -2)
            {
                Debug.WriteLine($"    ## New Session Detected ##");
                Task.Delay(10000).GetAwaiter().GetResult();
            }
            var rdpSession = (AxMsRdpClient9NotSafeForScripting)sender;
            rdpSession.Disconnect();
        }
        private void RdpConnectionOnOnDisconnected(object sender, IMsTscAxEvents_OnDisconnectedEvent e)
        {
            Environment.Exit(0);
        }
    }
}

在旁注中,我发现了一个问题,说可能有一种方法可以使用ActiveX控件(对于RDP)而不使用Windows窗体。我看到了他们给出的示例,但不确定是否可以在这种情况下使用他们的代码。

ActiveX control without a form

如果外面有人知道如何在不将ActiveX控件托管在窗体上的情况下执行此操作,请张贴示例。