rename.cpp 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439
  1. // lyfzRename.cpp : 定义控制台应用程序的入口点。
  2. //
  3. #include "stdafx.h"
  4. #include "rename.h"
  5. #include <Shlwapi.h>
  6. #include <io.h>
  7. #include "PARAMETERS.h"
  8. #ifdef _DEBUG
  9. #define new DEBUG_NEW
  10. #endif
  11. HMODULE g_hCurModule = NULL;
  12. TCHAR g_szCurModulePath[MAX_PATH] = {0};
  13. TCHAR g_szFna[MAX_PATH] = {0};
  14. BOOL GetDebugPriv();
  15. void WriteTextLog(const TCHAR *format, ...);
  16. string GetFileVersion(const string &strFilePath);
  17. bool GetFileVersion(IN const TCHAR *fname, IN HMODULE hModule, OUT WORD *pBuffer);
  18. bool GetProductVersion(IN const TCHAR *fname, IN HMODULE hModule, OUT WORD *pBuffer);
  19. void getopt(PARAMETERS & params, int argc, char ** argv);
  20. // 唯一的应用程序对象
  21. CWinApp theApp;
  22. using namespace std;
  23. int _tmain(int argc, TCHAR* argv[], TCHAR* envp[])
  24. {
  25. int nRetCode = 0;
  26. TCHAR szDrive[_MAX_DRIVE] = { 0 };
  27. TCHAR szDir[_MAX_DIR] = { 0 };
  28. TCHAR szFna[_MAX_DIR] = { 0 };
  29. TCHAR szExt[_MAX_DIR] = { 0 };
  30. ::GetModuleFileName(NULL, g_szCurModulePath, sizeof(g_szCurModulePath) / sizeof(TCHAR));
  31. _tsplitpath_s(g_szCurModulePath, szDrive, szDir, g_szFna, szExt);
  32. _tcscpy_s(g_szCurModulePath, szDrive);
  33. _tcscat_s(g_szCurModulePath, szDir);
  34. GetDebugPriv();
  35. // 初始化 MFC 并在失败时显示错误
  36. if (!AfxWinInit(::GetModuleHandle(NULL), NULL, ::GetCommandLine(), 0))
  37. {
  38. // TODO: 更改错误代码以符合您的需要
  39. _tprintf(_T("错误: MFC 初始化失败\n"));
  40. nRetCode = 1;
  41. }
  42. else
  43. {
  44. // 获取参数;
  45. PARAMETERS params;
  46. getopt(params, argc, argv);
  47. #ifdef _DEBUG
  48. WriteTextLog(_T("%ld, %ld,%s,%s"),params.dwProcId, params.killproc, params.NewVerPath.c_str(), params.OldVerPath.c_str());
  49. #endif
  50. if ( params.dwProcId != 0)
  51. {
  52. // 等待进程退出;
  53. DWORD dwTargetProId = 0;
  54. HANDLE hProc = OpenProcess(PROCESS_ALL_ACCESS, FALSE, params.dwProcId);
  55. if ( hProc != NULL)
  56. {
  57. if ( params.killproc )
  58. {//结束进程;
  59. TerminateProcess(hProc, 0);
  60. Sleep(500);
  61. }
  62. else
  63. {
  64. // 等待线程结束;
  65. WaitForSingleObject(hProc, INFINITE);
  66. CloseHandle(hProc);
  67. }
  68. }
  69. if (!PathFileExists(params.NewVerPath.c_str()))
  70. {
  71. exit(0);
  72. }
  73. #if 0//以主程序为升级包的处理方法;
  74. // 删除旧版本文件;
  75. DeleteFile(params.OldVerPath.c_str());
  76. // 移动文件;
  77. MoveFile(params.NewVerPath.c_str(), params.OldVerPath.c_str());
  78. if ( params.killproc )
  79. {
  80. // 启动程序;
  81. CoInitializeEx(NULL, COINIT_APARTMENTTHREADED | COINIT_DISABLE_OLE1DDE);//某些情况下要初始化COM才能使用ShellExecuteEx;
  82. SHELLEXECUTEINFO stuExecInfo = { 0 };
  83. DWORD dwExitCode = STILL_ACTIVE;
  84. stuExecInfo.cbSize = sizeof(SHELLEXECUTEINFO);
  85. stuExecInfo.lpFile = params.OldVerPath.c_str();
  86. stuExecInfo.nShow = SW_SHOW;
  87. stuExecInfo.lpVerb = _T("open");
  88. stuExecInfo.fMask = SEE_MASK_NOCLOSEPROCESS | SEE_MASK_FLAG_NO_UI;
  89. if (!ShellExecuteEx(&stuExecInfo))
  90. {
  91. DWORD dwError = GetLastError();
  92. WriteTextLog(_T("ShellExecuteEx Error=%ld"), dwError);
  93. return -1;
  94. }
  95. HANDLE hProcess = stuExecInfo.hProcess;
  96. if ( hProcess != NULL )
  97. {
  98. //WaitForSingleObject(hProcess,INFINITE);//不等待结束;
  99. CloseHandle(hProcess);
  100. }
  101. }
  102. #else//以安装包为升级包的处理方法;//可以选择静默安装安装包;
  103. if ( params.killproc )
  104. {
  105. // 启动程序;
  106. CoInitializeEx(NULL, COINIT_APARTMENTTHREADED | COINIT_DISABLE_OLE1DDE);//某些情况下要初始化COM才能使用ShellExecuteEx;
  107. SHELLEXECUTEINFO stuExecInfo = { 0 };
  108. DWORD dwExitCode = STILL_ACTIVE;
  109. CString strParameters = _T("/S");
  110. stuExecInfo.cbSize = sizeof(SHELLEXECUTEINFO);
  111. stuExecInfo.lpFile = params.NewVerPath.c_str();
  112. stuExecInfo.nShow = SW_SHOW;
  113. //stuExecInfo.lpParameters = strParameters;//不静默安装;
  114. stuExecInfo.lpVerb = _T("open");
  115. stuExecInfo.fMask = SEE_MASK_NOCLOSEPROCESS | SEE_MASK_FLAG_NO_UI;
  116. if (!ShellExecuteEx(&stuExecInfo))
  117. {
  118. DWORD dwError = GetLastError();
  119. WriteTextLog(_T("ShellExecuteEx Error=%ld"), dwError);
  120. return -1;
  121. }
  122. HANDLE hProcess = stuExecInfo.hProcess;
  123. if ( hProcess != NULL )
  124. {
  125. CloseHandle(hProcess);
  126. }
  127. }
  128. #endif
  129. }
  130. }
  131. return nRetCode;
  132. }
  133. void getopt(PARAMETERS & params, int argc, char ** argv)
  134. {
  135. std::string cmd = "";
  136. for ( argc > 0; argc--; argv++ )
  137. {
  138. if ( _tcsicmp(*argv, _T("--pid")) == 0)
  139. {
  140. cmd = _T("--pid");
  141. }
  142. else if (_tcsicmp(*argv, _T("--oldpath")) == 0)
  143. {
  144. cmd = _T("--oldpath");
  145. }
  146. else if (_tcsicmp(*argv, _T("--newpath")) == 0)
  147. {
  148. cmd = _T("--newpath");
  149. }
  150. else if (_tcsicmp(*argv, _T("--killproc")) == 0)
  151. {
  152. cmd = _T("--killproc");
  153. }
  154. else
  155. {
  156. if ( _tcsicmp(cmd.c_str(), _T("--pid")) == 0)
  157. {
  158. params.dwProcId = atoi(*argv);
  159. }
  160. else if (_tcsicmp(cmd.c_str(), _T("--oldpath")) == 0)
  161. {
  162. params.OldVerPath = *argv;
  163. }
  164. else if (_tcsicmp(cmd.c_str(), _T("--newpath")) == 0)
  165. {
  166. params.NewVerPath = *argv;
  167. }
  168. else if (_tcsicmp(cmd.c_str(), _T("--killproc")) == 0)
  169. {
  170. params.killproc = atoi(*argv);
  171. }
  172. cmd = "";
  173. }
  174. }
  175. }
  176. //2.获得其他exe或dll的版本信息
  177. string GetFileVersion(const string &strFilePath)
  178. {
  179. DWORD dwSize;
  180. DWORD dwRtn;
  181. string szVersion;
  182. //获取版本信息大小
  183. dwSize = GetFileVersionInfoSize(strFilePath.c_str(),NULL);
  184. DWORD dwErr = GetLastError();
  185. if (dwSize == 0)
  186. {
  187. return "";
  188. }
  189. char *pBuf;
  190. pBuf= new char[dwSize + 1];
  191. if(pBuf == NULL)
  192. return "";
  193. memset(pBuf, 0, dwSize + 1);
  194. //获取版本信息
  195. dwRtn = GetFileVersionInfo(strFilePath.c_str(),NULL, dwSize, pBuf);
  196. if(dwRtn == 0)
  197. {
  198. return "";
  199. }
  200. LPVOID lpBuffer = NULL;
  201. UINT uLen = 0;
  202. //版本资源中获取信息
  203. dwRtn = VerQueryValue(pBuf,
  204. TEXT("\\StringFileInfo\\080404b0\\FileVersion"), //0804中文
  205. //04b0即1252,ANSI
  206. //可以从ResourceView中的Version中BlockHeader中看到
  207. //可以测试的属性
  208. /*
  209. CompanyName
  210. FileDescription
  211. FileVersion
  212. InternalName
  213. LegalCopyright
  214. OriginalFilename
  215. ProductName
  216. ProductVersion
  217. Comments
  218. LegalTrademarks
  219. PrivateBuild
  220. SpecialBuild
  221. */
  222. &lpBuffer,
  223. &uLen);
  224. if(dwRtn == 0)
  225. {
  226. return "";
  227. }
  228. szVersion = (char*)lpBuffer;
  229. delete pBuf;
  230. return szVersion;
  231. }
  232. // hModule 模块句柄 NULL表示当前模块;
  233. bool GetFileVersion(IN const TCHAR *fname, IN HMODULE hModule, OUT WORD *pBuffer)
  234. {
  235. //TCHAR fname[MAX_PATH];
  236. VS_FIXEDFILEINFO *pVi = NULL;
  237. DWORD dwHandle = 0;
  238. string str;
  239. int size = GetFileVersionInfoSize(fname, &dwHandle);
  240. if (size > 0)
  241. {
  242. BYTE *buffer = new BYTE[size];
  243. memset(buffer,0,size);
  244. if (GetFileVersionInfo(fname, dwHandle, size, buffer))
  245. {
  246. if (VerQueryValue(buffer, _T("\\"), (LPVOID *)&pVi, (PUINT)&size))
  247. {
  248. pBuffer[0] = HIWORD(pVi->dwFileVersionMS);
  249. pBuffer[1] = LOWORD(pVi->dwFileVersionMS);
  250. pBuffer[2] = HIWORD(pVi->dwFileVersionLS);
  251. pBuffer[3] = LOWORD(pVi->dwFileVersionLS);
  252. delete []buffer;
  253. return true;
  254. }
  255. }
  256. delete []buffer;
  257. }
  258. return false;
  259. }
  260. bool GetProductVersion(IN const TCHAR *fname, IN HMODULE hModule, OUT WORD *pBuffer)
  261. {
  262. //TCHAR fname[MAX_PATH];
  263. VS_FIXEDFILEINFO *pVi = NULL;
  264. DWORD dwHandle = 0;
  265. string str;
  266. int size = GetFileVersionInfoSize(fname, &dwHandle);
  267. if (size > 0)
  268. {
  269. BYTE *buffer = new BYTE[size];
  270. memset(buffer,0,size);
  271. if (GetFileVersionInfo(fname, dwHandle, size, buffer))
  272. {
  273. if (VerQueryValue(buffer, _T("\\"), (LPVOID *)&pVi, (PUINT)&size))
  274. {
  275. pBuffer[0] = HIWORD(pVi->dwProductVersionMS);
  276. pBuffer[1] = LOWORD(pVi->dwProductVersionMS);
  277. pBuffer[2] = HIWORD(pVi->dwProductVersionLS);
  278. pBuffer[3] = LOWORD(pVi->dwProductVersionLS);
  279. delete []buffer;
  280. return true;
  281. }
  282. }
  283. delete []buffer;
  284. }
  285. return false;
  286. }
  287. BOOL GetDebugPriv()
  288. {
  289. // 返回的访问令牌指针;
  290. HANDLE hToken;
  291. // 接收所返回的制定特权名称的信息;
  292. LUID sedebugnameValue;
  293. // 新特权信息的指针(结构体);
  294. TOKEN_PRIVILEGES tkp;
  295. DWORD dwCurProcId = GetCurrentProcessId();
  296. // 要修改访问权限的进程句柄;
  297. HANDLE hCurProc;
  298. hCurProc = ::OpenProcess( PROCESS_ALL_ACCESS, FALSE, dwCurProcId );
  299. if ( !::OpenProcessToken( hCurProc,TOKEN_ADJUST_PRIVILEGES | TOKEN_QUERY, &hToken ) )
  300. {
  301. DWORD dwError = GetLastError();
  302. //CString strError;
  303. //strError.Format("升级包提权失败,错误码=%d",dwError);
  304. //AfxMessageBox(strError);
  305. return FALSE;
  306. }
  307. if ( ! ::LookupPrivilegeValue( NULL, SE_DEBUG_NAME, &sedebugnameValue ) )
  308. {
  309. DWORD dwError = GetLastError();
  310. //CString strError;
  311. /// strError.Format("升级包提权失败,错误码=%d",dwError);
  312. //AfxMessageBox(strError);
  313. CloseHandle( hToken );
  314. return FALSE;
  315. }
  316. tkp.PrivilegeCount = 1;
  317. tkp.Privileges[0].Luid = sedebugnameValue;
  318. tkp.Privileges[0].Attributes = SE_PRIVILEGE_ENABLED;
  319. if (!::AdjustTokenPrivileges( hToken, FALSE, &tkp, sizeof tkp, NULL, NULL ) )
  320. {
  321. DWORD dwError = GetLastError();
  322. //CString strError;
  323. //strError.Format("升级包提权失败,错误码=%d",dwError);
  324. //AfxMessageBox(strError);
  325. CloseHandle( hToken );
  326. return FALSE;
  327. }
  328. CloseHandle(hCurProc);
  329. CloseHandle(hToken);
  330. return TRUE;
  331. }
  332. /************************************************************************/
  333. /* 函数:WriteTextLog[7/28/2009 Jeff];
  334. /* 描述:写文本日志;
  335. /* 参数:;
  336. /* [IN] :;
  337. /* 返回:void;
  338. /* 注意:;
  339. /* 示例:;
  340. /*
  341. /* 修改:;
  342. /* 日期:;
  343. /* 内容:;
  344. /************************************************************************/
  345. void WriteTextLog(const TCHAR *format, ...)
  346. {
  347. // 解析出日志路径;
  348. TCHAR szlogpath[MAX_PATH] = { 0 };
  349. _stprintf_s(szlogpath, _T("%s%s.txt"), g_szCurModulePath, g_szFna);
  350. // 打开或创建文件;
  351. FILE *fp = NULL;
  352. //if (_taccess(szlogpath, 0) != -1)
  353. #ifndef UNICODE
  354. if (_access(szlogpath, 0) != -1)
  355. #else
  356. if (_taccess(szlogpath, 0) != -1)
  357. #endif
  358. {// 存在;
  359. fp = _tfopen(szlogpath, _T("a+"));
  360. // 移动到末尾;
  361. fseek(fp, 0, SEEK_END);
  362. }
  363. else
  364. {// 不存在;
  365. fp = _tfopen(szlogpath, _T("w+"));
  366. }
  367. if ( fp == NULL )
  368. return;
  369. // 格式化前设置语言区域;
  370. TCHAR* old_locale = _tcsdup(_tsetlocale(LC_CTYPE, NULL));
  371. _tsetlocale(LC_CTYPE, _T("chs"));//设定中文;
  372. // 格式化日志内容;
  373. va_list args = NULL;
  374. int len = 0;
  375. TCHAR *buffer = NULL;
  376. va_start(args, format);
  377. // _vscprintf doesn't count. terminating '\0'
  378. len = _vsctprintf(format, args) + 1;
  379. buffer = (TCHAR*)malloc(len * sizeof(TCHAR));
  380. _vstprintf_s(buffer, len, format, args);
  381. // 将日志内容输入到文件中;
  382. // 获取今年年份;
  383. __time64_t gmt = time(NULL);// 获取当前日历时间(1900-01-01开始的Unix时间戳);
  384. struct tm gmtm = {0};
  385. localtime_s(&gmtm, &gmt); // 时间戳转成本地时间;
  386. _ftprintf(fp, _T("%04d-%02d-%02d %02d:%02d:%02d %s\n"), gmtm.tm_year+1990, gmtm.tm_mon+1, gmtm.tm_mday, gmtm.tm_hour, gmtm.tm_min, gmtm.tm_sec, buffer);
  387. // 关闭文件,释放资源并设置回原语言区域;
  388. free(buffer);
  389. fclose(fp);
  390. _tsetlocale(LC_CTYPE, old_locale);
  391. free(old_locale);//还原区域设定;
  392. }