問: 您好,腳本專家!如何確定腳本正在哪一個(gè)帳戶下運(yùn)行? -- KW 答: 您好,KW。您知道,自從我們以各種托辭而開設(shè)這一專欄以來已有一段時(shí)間了,對于我們而言,這并非易事:畢竟,尋找托辭是我們這些腳本專家的拿手好戲。明確了這一點(diǎn),那就讓我們以我們最喜歡的一個(gè)托辭開始吧:我們將向您介紹的腳本只在 Windows XP 和 Windows Server 2003 上有效。我們將向您介紹使得該腳本在 Windows 2000 上同樣有效的方法,但后者絕對不及前者好。 噢,是的:現(xiàn)在感覺該方法不錯(cuò)。 好了,不找托辭了(至少是現(xiàn)在)。還是讓我們討論一下腳本吧。該腳本如下:
復(fù)制代碼 代碼如下:
strComputer = "." Set objWMIService = GetObject("winmgmts://" & strComputer & "/root/cimv2") Set colProcessList = objWMIService.ExecQuery("Select * from Win32_Process Where " & _ "Name = 'cscript.exe' or Name = 'wscript.exe'") For Each objProcess in colProcessList If InStr(objProcess.CommandLine, "test.vbs") Then colProperties = objProcess.GetOwner(strNameOfUser,strUserDomain) Wscript.Echo "This script is running under the account belonging to " _ & strUserDomain & "/" & strNameOfUser & "." End If Next
Set colProcessList = objWMIService.ExecQuery("Select * from Win32_Process Where " & _ "Name = 'cscript.exe' or Name = 'wscript.exe'")
您可能已經(jīng)猜到我們需要使用 Win32_Process 類來執(zhí)行我們的任務(wù),這是因?yàn)?nbsp;Win32_Process 是用來跟蹤計(jì)算機(jī)上當(dāng)前運(yùn)行的所有進(jìn)程的 WMI 類。當(dāng)然,我們并不關(guān)心計(jì)算機(jī)上運(yùn)行的所有進(jìn)程,我們只關(guān)心腳本。正因如此,我們添加了一個(gè) Where 子句,該子句將只返回以下兩個(gè) Windows 腳本宿主的實(shí)例的信息:Cscript.exe 和 Wscript.exe。 注意:是的,我們本來可以以稍有不同的方式來編寫此腳本的,或許那樣會(huì)在進(jìn)程中省下一兩行代碼。我們之所以選擇了此方法,是因?yàn)樵摲椒ㄅc我們在 Windows 2000 上執(zhí)行此任務(wù)的方法更相似。 發(fā)出查詢后,我們建立一個(gè) For Each 循環(huán),以遍歷返回的集合。在本例中,我們試圖確定名為 Test.vbs 的腳本的所有者。因此,我們需要檢查每個(gè)腳本,以查看它的名稱是否為 Test.vbs。我們?nèi)绾稳プ瞿兀客ㄟ^使用下面這行代碼: If InStr(objProcess.CommandLine, "test.vbs") Then 我們此處要做的是使用 InStr 函數(shù)來確定是否可在屬性 CommandLine 中的某個(gè)位置找到字符串 test.vbs。什么是 CommandLine 屬性?簡單地說,它就是從命令提示符啟動(dòng)腳本所需的命令字符串。例如,CommandLine 可能為下列內(nèi)容: C:/Scripts/Test Scripts/Test.vbs 由于我們假定不存在名為 MyTest.vbs 的腳本,因此我們將檢查 test.vbs。如果您擔(dān)心此類名稱沖突,那么,我們可以只使用 InStr 并針對類似 /test.vbs 的字符串進(jìn)行測試。這是一個(gè)您必須決定的問題。 如果確實(shí)可以在 CommandLine 值中找到我們的目標(biāo)字符串,則我們將調(diào)用 GetOwner 方法來找出進(jìn)程的“所有者”(即,腳本在其下運(yùn)行的帳戶的名稱): objProcess.GetOwner(strNameOfUser,strUserDomain) 我們需要使用 GetOwner 傳遞一對“輸出參數(shù)”。輸出參數(shù)就是方法將用一個(gè)值對其進(jìn)行填充的變量(由我們自己來命名該變量)。這里,我們將傳遞名為 strNameOfUser 和 strUserDomain 的變量。反過來,GetOwner 將用戶名稱和擁有進(jìn)程的用戶所在的域賦值給這兩個(gè)變量。 此時(shí)我們所要做的就是回顯返回信息: Wscript.Echo "This script is running under the account belonging to " _ & strUserDomain & "/" & strNameOfUser & "." 那么,我們?yōu)楹尾荒茉?nbsp;Windows 2000 上運(yùn)行此腳本呢?實(shí)際上,有充分的理由來對此進(jìn)行解釋:只有 Windows XP 和 Windows Server 2003 上才有 CommandLine 屬性。在其他版本的 Windows 上,我們無法標(biāo)識(shí)各個(gè)腳本;最好的方法就是為恰好正在運(yùn)行的 Cscript.exe 和 Wscript.exe 的所有實(shí)例返回所有者信息。如果只有一個(gè)腳本在運(yùn)行,也同樣沒有問題:CScript.exe 或 Wscript.exe 的單個(gè)實(shí)例必須為該單個(gè)腳本。換句話說,這就意味著腳本宿主進(jìn)程的所有者也是腳本進(jìn)程的所有者。如果運(yùn)行了多個(gè)腳本,則是另外一回事了。如果確實(shí)為此種情況,您最好是說:“嗯,Ken Myer 擁有其中的一個(gè)腳本,盡管我們不知道具體是哪一個(gè)。他未擁有的某個(gè)腳本恰好為 Pilar Ackerman 所擁有。 不,沒有那么好。不過事實(shí)就是這樣。(是的,這是一個(gè)托辭。盡管有點(diǎn)漏洞百出,但它仍是一個(gè)托辭。) 下面是 Windows 2000 解決方案(也可以說是:部分解決方案):
復(fù)制代碼 代碼如下:
strComputer = "." Set objWMIService = GetObject("winmgmts://" & strComputer & "/root/cimv2") Set colProcessList = objWMIService.ExecQuery("Select * from Win32_Process Where " & _ "Name = 'cscript.exe' or Name = 'wscript.exe'") For Each objProcess in colProcessList objProcess.GetOwner strNameOfUser,strUserDomain Wscript.Echo "A script is running under the account belonging to " _ & strUserDomain & "/" & strNameOfUser & "." Next