淺談Python和C#中的Run As代碼實現(xiàn)方式
1,引入
最近因為項目的原因,需要在自動化測試代碼中實現(xiàn)用戶角色的切換,自然第一印象就想到了Run As(中文版windows中也叫“運行方式”)。比如我們可以在打開IE瀏覽器的時候右鍵單擊“運行方式”:
然后輸入一個新的用戶帳戶,如下圖:
上圖中我本機使用的是域帳戶本地帳戶登陸的操作系統(tǒng),但是我現(xiàn)在需要使用域帳戶來運行IE瀏覽器,以方便我在打開相應的綁定域帳戶權限的頁面,如公司內(nèi)部站點。這樣就不用在打開相應的頁面時候輸入域帳戶名了。
有些時候我們也可以應用相同的方式來使用管理員帳戶啟動相應的應用程序:
在較近操作系統(tǒng)中,右鍵菜單中沒有了“Run as”,而是變成了“Run As Administrator”,目的是提高運行相應程序的用戶權限:
好了,言歸正傳,今天筆者提到的則是使用代碼來實現(xiàn)相同的功能,即以更改運行程序的用戶帳戶。
2,代碼實現(xiàn)命令行Run As
命令行
用命令行實現(xiàn)上述的Run as功能大致可以這樣寫:
runas /user:user@domain.com "C:\Program Files\Internet Explorer\iexplore.exe"
Python實現(xiàn)
RunAsWithinPython.py文件內(nèi)容如下:
-->import os \"C:\Program Files\Internet Explorer\iexplore.exe\"") |
額,就這兩行,看起來比命令行內(nèi)容多不了幾個字符,不得不驚詫Python的強悍。另外,python使用縮進來表示代碼層次而不是我們在C#等語言中用的大括號“{”,代碼看起來很爽,當然,這是閑話了。
C#實現(xiàn)
Program.cs文件內(nèi)容如下:
using System.Diagnostics; |
恩,C#表現(xiàn)也不錯,看來確實是一個比較簡單的命令而已,如果你沒有運行一下這兩個程序,你可能以為這樣就搞定了,相當簡單。但是如果你是一個細心的人,或者你是用過runas命令行的話,你應該知道,這里有問題:
很可惜的是,這兩種方式存在一個致命的缺陷:只能在第一個命令行中輸入用戶名,然后才會由系統(tǒng)提示輸入密碼,這樣就不能在一條命令行中將用戶名和密碼完全包含。這樣我們只能使用代碼實現(xiàn)一半,另外一半則仍然需要手動輸入(密碼)——顯然這就造成了一個半拉子工程。我們也可以查看一下runas幫助說明:
這樣設計的目的在于防止像我們這種弱弱的想要使用命令行來提高運行程序的權限的人——至少我們是不能使用批處理文件來干這件事了,這樣更安全了,不是嗎?好吧,接下來,我們將使用另外一種方法,這種方法則不會出現(xiàn)上述半拉子工程的尷尬情況。
3, 使用Process的ProcessStartInfo實現(xiàn)
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Diagnostics;
namespace MainTest
{
class ProcessUserLogonHelper
{
///
/// 在Main函數(shù)中調(diào)用該方法的的示例代碼
///
private void InvokeExample()
{
ProcessUserLogonHelper phelper = new ProcessUserLogonHelper();
Process pro = phelper.CreateProcessWithUserToken(@"C:\Program Files\Internet Explorer\iexplore.exe", "bestreme.com", "chdwu", "************");
pro.Start();
}
///
/// 使用指定帳戶綁定進程
///
/// 進程的宿主程序
/// 指定帳戶的域名
/// 指定帳戶的用戶名
/// 指定帳戶的密碼
///綁定了特定帳戶的進程
public Process CreateProcessWithUserToken(string appPath,string domain, string userName, string password)
{
Process pro = new Process();
ProcessStartInfo processInfo = new ProcessStartInfo(appPath);
processInfo.UseShellExecute = false;
processInfo.UserName = userName;
processInfo.Domain = domain;
System.Security.SecureString psw = new System.Security.SecureString();
foreach (char c in password.ToCharArray())
{
psw.AppendChar(c);
}
processInfo.Password = psw;
processInfo.WorkingDirectory = System.IO.Path.GetDirectoryName(appPath);
pro.StartInfo = processInfo;
return pro;
}
///
/// 使用指定帳戶綁定進程
///
/// 進程的宿主程序
/// 指定本地帳戶的用戶名
/// 指定帳戶的密碼
///綁定了特定帳戶的進程
public Process CreateProcessWithUserToken(string appPath, string userName, string password)
{
return CreateProcessWithUserToken(appPath, "", userName, password);
}
}
}
上面的代碼有幾個地方需要指出的是:
1,應該待運行的程序指定相應的啟動路徑,否則的話在上述代碼運行的過程中可能提示“目錄不存在”;
2,Domain可以不設置,而直接使用bestreme\chdwu的方式給UserName賦值即可;
3,UseShellExecute 應該設置為False
4,ProcessStartInfo中的password是System.Security.SecureString類型的,所以如果我們只是傳入了一個string作為password,我們應該將其轉(zhuǎn)換為正確的類型。
結束語
上面為大家介紹了幾種Run as的實現(xiàn)方式,其中的前面兩種分別用Python和C#實現(xiàn)了命令行Run as的功能,第三種方法則是在ProcessStartInfo中加入了用戶信息以實現(xiàn)用指定用戶啟動運行應用程序。
當然實現(xiàn)Run as的代碼實現(xiàn)方法很多,在下一篇中將介紹一下有關使用Win32 API的實現(xiàn)。需要指出的是,使用Win32 API幾種實現(xiàn)方式存在著一些潛在的問題,詳細內(nèi)容將盡快在下一篇文章中給大家展示。
【編輯推薦】