以不同用户身份运行程序,/savecred只需要输入一次密码(GetTokenByName取得EXPLORER.EXE的令牌,然后调用CreateProcessAsUser,而且使用LoadUserProfile解决另存文件的问题)good

时间:2024-04-29 08:34:48

http://blog.sina.com.cn/s/blog_65977dde0100s7tm.html

--------------------------------------------------

以不同用户身份运行程序

(2011-07-09 09:15:25)
一直想方便的处理CCProxy代理的帐号管理,所以梦想做一个比较好的管理工具。但一个最麻烦的问题就是帐号的更新,CCProxy有一个网页管理功 能,可以加帐号,但加的帐号就是不可以立即更新。中午上网的时候发现CCProxy有一功能就是支持命令行的操作,如:
  CCProxy -reboot  重启软件
  CCProxy -reset   更新配置
  CCProxy -update  更新帐号
   
   试着改了AccInfo.ini中帐号信息,在DOS中运行CCProxy -update的确更新了账号,所以开始用PHP做管理工具,做到调用CCProxy -update时,用了PHP中的exec(),system()等函数一直没有效果,后又通过调用批处理文件来调用命令行参数都不行。 处理得正没耐心的时候,一气之下狂刷新PHP网页,电脑卡死,用进程管理器查看时发现打开了多个CCProxy进程,认真一看,除了一个CCProxy是用户进程外其它CCProxy全是system进程。认真一想有可能是运行用户身份不同所产生的结果。
  Apache服务调用的外部程序以system身份运行,自己双击运行的程序以用户身份运行。 如 果CCProxy -update以用户身份运行是不是就可以了呢?本人在网络上找到了runas这个命令,的确可以指定以哪个用户运行,但是每次都要输密码,没有密码的帐号就要加上密码才可以用,“/savecred”这个参数可以用,只要输入一次密码就可以了,但在PHP中发现要以system的身份输入一次才行,根本 没有机会输入。打算用C程序来处理这个问题。可是发现用WinExec(),ShellExecute(),CreateProcess()都不好处理这个问题,好在发现了CreateProcessAsUser()这个函数。把网络上的程序改了几处,编译后一试问题终于解决。
   以下为相关代码:
// Update.cpp : 定义控制台应用程序的入口点。
  1. #include "stdafx.h"
  2. #include <windows.h>
  3. #include <tlhelp32.h>
  4. BOOL GetTokenByName(HANDLE &hToken,LPSTR lpName)
  5. {
  6. if(!lpName)
  7. return FALSE;
  8. HANDLE         hProcessSnap = NULL;
  9. BOOL           bRet      = FALSE;
  10. PROCESSENTRY32 pe32      = {0};
  11. hProcessSnap = CreateToolhelp32Snapshot(TH32CS_SNAPPROCESS, 0);
  12. if (hProcessSnap == INVALID_HANDLE_VALUE)
  13. return (FALSE);
  14. pe32.dwSize = sizeof(PROCESSENTRY32);
  15. if (Process32First(hProcessSnap, &pe32))
  16. {
  17. do
  18. {
  19. if(!strcmp(_strupr(pe32.szExeFile),_strupr(lpName)))  // 改成大写
  20. {
  21. HANDLE hProcess = OpenProcess(PROCESS_QUERY_INFORMATION,
  22. FALSE,pe32.th32ProcessID);
  23. bRet = OpenProcessToken(hProcess,TOKEN_ALL_ACCESS,&hToken);
  24. CloseHandle (hProcessSnap);
  25. return (bRet);
  26. }
  27. }
  28. while (Process32Next(hProcessSnap, &pe32));
  29. bRet = TRUE;
  30. }
  31. else
  32. bRet = FALSE;
  33. CloseHandle (hProcessSnap);
  34. return (bRet);
  35. }
  36. BOOL RunProcess(LPCSTR lpImage,LPSTR lpCommandLine)
  37. {
  38. if(!lpImage)
  39. return FALSE;
  40. HANDLE hToken;
  41. if(!GetTokenByName(hToken,"EXPLORER.EXE"))
  42. return FALSE;
  43. STARTUPINFO si;
  44. PROCESS_INFORMATION pi;
  45. ZeroMemory(&si, sizeof(STARTUPINFO));
  46. si.cb= sizeof(STARTUPINFO);
  47. si.lpDesktop = TEXT("winsta0\\default");
  48. BOOL bResult = CreateProcessAsUser(hToken,lpImage, lpCommandLine,NULL,NULL,
  49. FALSE,NORMAL_PRIORITY_CLASS,NULL,NULL,&si,&pi);
  50. CloseHandle(hToken);
  51. if(bResult)
  52. {
  53. OutputDebugString("CreateProcessAsUser ok!\r\n");
  54. printf("CreateProcessAsUser ok!\r\n");
  55. }
  56. else
  57. {
  58. OutputDebugString("CreateProcessAsUse* **lse!\r\n");
  59. printf("CreateProcessAsUse* **lse!\r\n");
  60. }
  61. return bResult;
  62. }
  63. int _tmain(int argc, _TCHAR* argv[])
  64. {
  65. RunProcess("CCProxy.exe"," -update");
  66. return 0;
  67. }
 http://www.cnblogs.com/-clq/archive/2012/01/19/2326263.html
3楼 likang0712 2012-02-02 11:58发表 [回复]
谢谢 :)

2楼 santalence 2009-05-23 17:29发表 [回复]
这个方法确实很好,to aguiwang :解决 GetSaveFileName 的问题,可在调用CreateProcessAsUser前调用下面这个函数:
BOOL LoadProfile(HANDLE hToken, LPTSTR lpszUserName)
{
BOOL bRet = FALSE;

PROFILEINFO pi;
pi.dwSize = sizeof(PROFILEINFO);
pi.dwFlags = 0;
pi.lpUserName = lpszUserName;
pi.lpDefaultPath = NULL;
pi.lpPolicyPath = NULL;
pi.lpServerName = NULL;
pi.lpProfilePath = NULL;

bRet = ::LoadUserProfile(hToken, &pi);

return bRet;
}

其中的lpszUsername 就是当前登陆用户名

1楼 aguiwang 2006-08-30 09:00发表 [回复]
idAnts的方法很好,但这样从服务里用CreateProcessAsUser创建的进程,还是有调用者的性质.比如在创建出来的进程里弹出Save file的common dialog是(即调用API GetSaveFileName),上面的桌面和我的文件夹还是LocalService下的,而不是当前登陆用户下的.而Localservice又没有桌面和我的文件夹目录,所以在选择桌面和我的文件夹时common dialog会出现这样的错误提示错误.C:/documents and settings/localservice/desktop refers to a location that is unavailable. It could be on a hard drive on this computer,on a network, or on a different computer on your home network. Check to make sure that the disk is properly inserted, or that you are connected to the Internet or home network, and then try again.