悲情土仔一生

  C++博客 :: 首页 :: 新随笔 :: 联系 :: 聚合  :: 管理 ::
  14 随笔 :: 0 文章 :: 74 评论 :: 0 Trackbacks

[翻译]利用C#获取终端服务(Terminal Services)会话的闲置时间

作者:Tuuzed(土仔)   发表于:2008年2月29日
版权声明:可以任意转载,转载时请务必以超链接形式标明文章原始出处和作者信息及本声明
http://www.cppblog.com/tuuzed/archive/2008/02/29/43424.html


 
原著:Guy Teverovsky
翻译:土仔Tuuzed
原文出处:Querying TS session idle time with C#
原文URL:http://blogs.microsoft.co.il/blogs/guyt/archive/2007/10/06/querying-ts-session-idle-time-with-c.aspx
本文特别声明:原文版权归原作者Guy Teverovsky所有,本译文可以任意转载, 转载时请务必以超链接形式标明原文出处、译文出处、译者及本声明。

        这一开始就是一个挑战。我的同事正在编写一个程序,用来获取远程计算机终端服务会话的闲置时间。刚开始,我们想到的是利用WMI(译注WMI:Windows Management Instrumentation管理规范),但是那里没有有价值的信息,因此我们开始在Win32 API(译注API:Application Programming Interface应用程序接口)里查找。首先,“Terminal Services API”字样令我们觉得有希望,同时里面的WTSQuerySessionInformation函数也引起了我的注意,可是,接下来对WTS_INFO_CLASS枚举类型的注释不得不使得我们又要回到Google中了:

 

        后来,我找到一个看似有些意思的API函数:WinStationQueryInformationW. 可是这个函数有些问题:

        1. 这是一个内部函数,它在Winsta.dll内部实现,不像其他已公开的WTS API函数是在wtsapi32.dll内部;
        2. 该函数的返回值(一个结构):WINSTATIONQUERYINFORMATIONW,该结构的代码在MSDN上没有任何资料;
        3. MSDN上的资料说,查询的句柄(handle)通常只能是SERVERNAME_CURRENT。这就意味着你不能查询远程的终端服务器(尽管如此,通过用depends.exe(译注:Visual Studio自带的工具)查看终端服务管理器(Terminal Services Manager)的EXE文件,该管理器也大量地在使用这个函数)。

       我决定试一试,看能否解决以上的问题:

       1. “这是一个内部函数”——嗯……没办法,已经没有其他方法能够得到我想要的信息了,这是唯一的选择。
       2. “WINSTATIONQUERYINFORMATIONW结构的代码没有公开”——这是最棘手的问题。看了它在Platform SDK的winternl.h文件中的定义后,我相当失望:


            显然,这是不够的……我设法找到了这部分结构的C++代码,转用C#定义如下:

       3. “MSDN上的资料说,我们只能在本地机器上使用这个函数”——但是这里隐含了些线索……这个函数与WTS API记载在了一起,而且你可以通过一个函数获得终端服务(TS)的句柄(handle):WTSOpenServer。所以,接下来我要做的就是利用WTSOpenServer()函数获取远程终端服务的句柄(handle),然后利用WinStationQueryInformationW()函数得到我想要的信息。

 

      把上面的做一个总述

      需要的结构(structures)/枚举类型(enums):


        DLL文件导入(DLLImports):

 


         我们将得到通俗易懂的信息:

 


        将难处理的FILETIME类型转为DateTime类型:

 


       有用的代码:

 


        只需简单地调用WTSQuerySessionInfo("servername", <Session ID>)就行了。

 

       总结:

       就如大家看到的一样,在所有步骤完成后,我们还是没有得到我们想要的“会话闲置时间(Session Idle Time)”。其实,根据下面的简单规则就能很容易地把它计算出来:

※ 如果会话是断开(disconnected)状态,闲置时间=当前时间-断开时间(Idle Time = CurrentTime - DisconnectTime)
※ 如果会话是活动的(alive)状态,闲置时间=当前时间-最后输入时间(Idle Time = CurrentTime - LastInputTime)


posted on 2008-02-29 01:07 土仔 阅读(4223) 评论(2)  编辑 收藏 引用 所属分类: 土仔编程

评论

# re: [翻译]利用C#获取终端服务(Terminal Services)会话的闲置时间 2008-12-29 16:21 北京时间
断开时间  回复  更多评论
  

# re: [翻译]利用C#获取终端服务(Terminal Services)会话的闲置时间 2011-03-03 18:49 Binyao
我用你的代码在2008R2上面运行,但 ret 永远都不=1啊,为什么?急啊,我的邮箱是v-bxu@microsoft.com
msn: xbyhoo@hotmail.com

ret = WinStationQueryInformationW(
hServer,
sessionId,
(uint)WINSTATIONINFOCLASS.WinStationInformation,
pwsInfo,
(uint)Marshal.SizeOf(typeof(WINSTATIONINFORMATIONW)),
ref RetLen);

if (ret == 1)
{
wsInfo = (WINSTATIONINFORMATIONW)Marshal.PtrToStructure(
pwsInfo,
typeof(WINSTATIONINFORMATIONW));
WTSCloseServer(hServer);
RetWsInfo.ConnectTime = FileTimeToDateTime(wsInfo.ConnectTime);
RetWsInfo.CurrentTime = FileTimeToDateTime(wsInfo.CurrentTime);
RetWsInfo.DisconnectTime = FileTimeToDateTime(wsInfo.DisconnectTime);
RetWsInfo.LastInputTime = FileTimeToDateTime(wsInfo.LastInputTime);
RetWsInfo.LoginTime = FileTimeToDateTime(wsInfo.LoginTime);
RetWsInfo.SessionId = (int)wsInfo.SessionId;
}  回复  更多评论
  


只有注册用户登录后才能发表评论。
网站导航: 博客园   IT新闻   BlogJava   知识库   博问   管理