mirror of
https://github.com/reactos/reactos.git
synced 2025-01-04 21:38:43 +00:00
350 lines
9.7 KiB
C
350 lines
9.7 KiB
C
/*
|
|
* PROJECT: ReactOS runas utility
|
|
* LICENSE: GPL - See COPYING in the top level directory
|
|
* FILE: base/applications/runas/runas.c
|
|
* COPYRIGHT: Copyright 2022 Eric Kohl <eric.kohl@reactos.org>
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <limits.h>
|
|
#include <stdarg.h>
|
|
|
|
#define WIN32_NO_STATUS
|
|
#include <windef.h>
|
|
#include <winbase.h>
|
|
#include <winnls.h>
|
|
#include <wincon.h>
|
|
#include <winsvc.h>
|
|
#include <conutils.h>
|
|
|
|
#include "resource.h"
|
|
|
|
#define NDEBUG
|
|
#include <debug.h>
|
|
|
|
#define MAX_PASSWORD_LENGTH 64
|
|
|
|
static
|
|
VOID
|
|
Usage(VOID)
|
|
{
|
|
int i;
|
|
for (i = IDS_USAGE01; i <= IDS_USAGE_MAX; i++)
|
|
ConResPuts(StdOut, i);
|
|
}
|
|
|
|
|
|
static
|
|
VOID
|
|
ConInString(
|
|
_In_ PWSTR pInput,
|
|
_In_ DWORD dwLength)
|
|
{
|
|
DWORD dwOldMode;
|
|
DWORD dwRead = 0;
|
|
HANDLE hFile;
|
|
PWSTR p;
|
|
PCHAR pBuf;
|
|
|
|
pBuf = (PCHAR)HeapAlloc(GetProcessHeap(), 0, dwLength - 1);
|
|
ZeroMemory(pInput, dwLength * sizeof(WCHAR));
|
|
hFile = GetStdHandle(STD_INPUT_HANDLE);
|
|
GetConsoleMode(hFile, &dwOldMode);
|
|
|
|
SetConsoleMode(hFile, ENABLE_LINE_INPUT /*| ENABLE_ECHO_INPUT*/);
|
|
|
|
ReadFile(hFile, (PVOID)pBuf, dwLength - 1, &dwRead, NULL);
|
|
|
|
MultiByteToWideChar(GetConsoleCP(), 0, pBuf, dwRead, pInput, dwLength - 1);
|
|
HeapFree(GetProcessHeap(), 0, pBuf);
|
|
|
|
for (p = pInput; *p; p++)
|
|
{
|
|
if (*p == L'\x0d')
|
|
{
|
|
*p = UNICODE_NULL;
|
|
break;
|
|
}
|
|
}
|
|
|
|
SetConsoleMode(hFile, dwOldMode);
|
|
}
|
|
|
|
|
|
int
|
|
wmain(
|
|
int argc,
|
|
LPCWSTR argv[])
|
|
{
|
|
LPCWSTR pszArg;
|
|
int i, result = 0;
|
|
BOOL bProfile = FALSE, bNoProfile = FALSE;
|
|
BOOL bEnv = FALSE, bNetOnly = FALSE;
|
|
PWSTR pszUserName = NULL;
|
|
PWSTR pszDomain = NULL;
|
|
PWSTR pszCommandLine = NULL;
|
|
PWSTR pszPassword = NULL;
|
|
PWSTR pszCurrentDirectory = NULL;
|
|
PWSTR pszEnvironment = NULL;
|
|
PWSTR ptr;
|
|
STARTUPINFOW StartupInfo;
|
|
PROCESS_INFORMATION ProcessInfo;
|
|
DWORD dwLogonFlags = LOGON_WITH_PROFILE;
|
|
DWORD dwCreateFlags = 0;
|
|
BOOL rc;
|
|
|
|
/* Initialize the Console Standard Streams */
|
|
ConInitStdStreams();
|
|
|
|
if (argc == 1)
|
|
{
|
|
Usage();
|
|
return 0;
|
|
}
|
|
|
|
ZeroMemory(&StartupInfo, sizeof(StartupInfo));
|
|
ZeroMemory(&ProcessInfo, sizeof(ProcessInfo));
|
|
|
|
for (i = 1; i < argc; i++)
|
|
{
|
|
pszArg = argv[i];
|
|
if (*pszArg == L'-' || *pszArg == L'/')
|
|
{
|
|
pszArg++;
|
|
if (wcscmp(pszArg, L"?") == 0)
|
|
{
|
|
Usage();
|
|
result = 0;
|
|
goto done;
|
|
}
|
|
else if (wcsicmp(pszArg, L"profile") == 0)
|
|
{
|
|
bProfile = TRUE;
|
|
}
|
|
else if (wcsicmp(pszArg, L"netonly") == 0)
|
|
{
|
|
bNetOnly = TRUE;
|
|
}
|
|
else if (wcsicmp(pszArg, L"noprofile") == 0)
|
|
{
|
|
bNoProfile = TRUE;
|
|
}
|
|
else if (wcsicmp(pszArg, L"env") == 0)
|
|
{
|
|
bEnv = TRUE;
|
|
}
|
|
else if (_wcsnicmp(pszArg, L"user:", 5) == 0)
|
|
{
|
|
pszArg += 5;
|
|
ptr = wcschr(pszArg, L'@');
|
|
if (ptr != NULL)
|
|
{
|
|
/* User@Domain */
|
|
pszUserName = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, ((ptr - pszArg) + 1) * sizeof(WCHAR));
|
|
if (pszUserName == NULL)
|
|
{
|
|
ConResPrintf(StdOut, IDS_INTERNAL_ERROR, ERROR_OUTOFMEMORY);
|
|
result = -1;
|
|
goto done;
|
|
}
|
|
|
|
wcsncpy(pszUserName, pszArg, (ptr - pszArg));
|
|
|
|
ptr++;
|
|
pszDomain = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, (wcslen(ptr) + 1) * sizeof(WCHAR));
|
|
if (pszDomain == NULL)
|
|
{
|
|
ConResPrintf(StdOut, IDS_INTERNAL_ERROR, ERROR_OUTOFMEMORY);
|
|
result = -1;
|
|
goto done;
|
|
}
|
|
|
|
wcscpy(pszDomain, ptr);
|
|
}
|
|
else
|
|
{
|
|
ptr = wcschr(pszArg, L'\\');
|
|
if (ptr != NULL)
|
|
{
|
|
/* Domain\User */
|
|
pszUserName = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, (wcslen(ptr + 1) + 1)* sizeof(WCHAR));
|
|
if (pszUserName == NULL)
|
|
{
|
|
ConResPrintf(StdOut, IDS_INTERNAL_ERROR, ERROR_OUTOFMEMORY);
|
|
result = -1;
|
|
goto done;
|
|
}
|
|
|
|
wcscpy(pszUserName, (ptr + 1));
|
|
|
|
pszDomain = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, ((ptr - pszArg) + 1) * sizeof(WCHAR));
|
|
if (pszDomain == NULL)
|
|
{
|
|
ConResPrintf(StdOut, IDS_INTERNAL_ERROR, ERROR_OUTOFMEMORY);
|
|
result = -1;
|
|
goto done;
|
|
}
|
|
|
|
wcsncpy(pszDomain, pszArg, (ptr - pszArg));
|
|
}
|
|
else
|
|
{
|
|
/* User */
|
|
pszUserName = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, (wcslen(pszArg) + 1) * sizeof(WCHAR));
|
|
if (pszUserName == NULL)
|
|
{
|
|
ConResPrintf(StdOut, IDS_INTERNAL_ERROR, ERROR_OUTOFMEMORY);
|
|
result = -1;
|
|
goto done;
|
|
}
|
|
|
|
wcscpy(pszUserName, pszArg);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
Usage();
|
|
result = -1;
|
|
goto done;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (pszCommandLine == NULL)
|
|
{
|
|
pszCommandLine = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, (wcslen(pszArg) + 1) * sizeof(WCHAR));
|
|
if (pszCommandLine == NULL)
|
|
{
|
|
ConResPrintf(StdOut, IDS_INTERNAL_ERROR, ERROR_OUTOFMEMORY);
|
|
result = -1;
|
|
goto done;
|
|
}
|
|
|
|
wcscpy(pszCommandLine, pszArg);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Check for incompatible options */
|
|
if ((bProfile && bNoProfile) ||
|
|
(bProfile && bNetOnly))
|
|
{
|
|
Usage();
|
|
result = -1;
|
|
goto done;
|
|
}
|
|
|
|
/* Check for existing command line and user name */
|
|
if (pszCommandLine == NULL || pszUserName == NULL)
|
|
{
|
|
Usage();
|
|
result = -1;
|
|
goto done;
|
|
}
|
|
|
|
if (bProfile)
|
|
dwLogonFlags |= LOGON_WITH_PROFILE;
|
|
|
|
if (bNoProfile)
|
|
dwLogonFlags &= ~LOGON_WITH_PROFILE;
|
|
|
|
if (bNetOnly)
|
|
{
|
|
dwLogonFlags |= LOGON_NETCREDENTIALS_ONLY;
|
|
dwLogonFlags &= ~LOGON_WITH_PROFILE;
|
|
}
|
|
|
|
DPRINT("User: %S\n", pszUserName);
|
|
DPRINT("Domain: %S\n", pszDomain);
|
|
DPRINT("CommandLine: %S\n", pszCommandLine);
|
|
|
|
if (pszDomain == NULL)
|
|
{
|
|
DWORD dwLength = MAX_COMPUTERNAME_LENGTH + 1;
|
|
pszDomain = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, dwLength * sizeof(WCHAR));
|
|
if (pszDomain == NULL)
|
|
{
|
|
ConResPrintf(StdOut, IDS_INTERNAL_ERROR, ERROR_OUTOFMEMORY);
|
|
result = -1;
|
|
goto done;
|
|
}
|
|
|
|
GetComputerNameW(pszDomain, &dwLength);
|
|
}
|
|
|
|
if (bEnv)
|
|
{
|
|
pszEnvironment = GetEnvironmentStringsW();
|
|
pszCurrentDirectory = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, (MAX_PATH + 1) * sizeof(WCHAR));
|
|
if (pszCurrentDirectory == NULL)
|
|
{
|
|
ConResPrintf(StdOut, IDS_INTERNAL_ERROR, ERROR_OUTOFMEMORY);
|
|
result = -1;
|
|
goto done;
|
|
}
|
|
|
|
GetCurrentDirectory(MAX_PATH + 1, pszCurrentDirectory);
|
|
dwCreateFlags |= CREATE_UNICODE_ENVIRONMENT;
|
|
}
|
|
|
|
pszPassword = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, (MAX_PASSWORD_LENGTH + 1) * sizeof(WCHAR));
|
|
if (pszPassword == NULL)
|
|
{
|
|
ConResPrintf(StdOut, IDS_INTERNAL_ERROR, ERROR_OUTOFMEMORY);
|
|
result = -1;
|
|
goto done;
|
|
}
|
|
|
|
/* Query the password */
|
|
ConResPrintf(StdOut, IDS_PASSWORD, pszDomain, pszUserName);
|
|
ConInString(pszPassword, MAX_PASSWORD_LENGTH + 1);
|
|
ConPuts(StdOut, L"\n");
|
|
|
|
ConResPrintf(StdOut, IDS_START, pszCommandLine, pszDomain, pszUserName);
|
|
|
|
rc = CreateProcessWithLogonW(pszUserName,
|
|
pszDomain,
|
|
pszPassword,
|
|
dwLogonFlags,
|
|
NULL,
|
|
pszCommandLine,
|
|
dwCreateFlags,
|
|
pszEnvironment,
|
|
pszCurrentDirectory,
|
|
&StartupInfo,
|
|
&ProcessInfo);
|
|
if (rc == FALSE)
|
|
{
|
|
ConResPrintf(StdOut, IDS_RUN_ERROR, pszCommandLine);
|
|
ConPrintf(StdOut, L"%lu\n", GetLastError());
|
|
}
|
|
|
|
done:
|
|
if (ProcessInfo.hThread)
|
|
CloseHandle(ProcessInfo.hThread);
|
|
|
|
if (ProcessInfo.hProcess)
|
|
CloseHandle(ProcessInfo.hProcess);
|
|
|
|
if (pszPassword)
|
|
HeapFree(GetProcessHeap(), 0, pszPassword);
|
|
|
|
/* NOTE: Do NOT free pszEnvironment */
|
|
|
|
if (pszCurrentDirectory)
|
|
HeapFree(GetProcessHeap(), 0, pszCurrentDirectory);
|
|
|
|
if (pszCommandLine)
|
|
HeapFree(GetProcessHeap(), 0, pszCommandLine);
|
|
|
|
if (pszUserName)
|
|
HeapFree(GetProcessHeap(), 0, pszUserName);
|
|
|
|
if (pszDomain)
|
|
HeapFree(GetProcessHeap(), 0, pszDomain);
|
|
|
|
return result;
|
|
}
|