#include "users.h" #include "util/windows/unicode.h" #include #include #include static inline uint64_t to_ms(uint64_t ret) { ret -= 116444736000000000ull; return ret / 10000ull; } const char* ffDetectUsers(FFUsersOptions* options, FFlist* users) { WTS_SESSION_INFO_1W* sessionInfo; DWORD sessionCount; DWORD level = 1; if (!WTSEnumerateSessionsExW(WTS_CURRENT_SERVER_HANDLE, &level, 0, &sessionInfo, &sessionCount)) return "WTSEnumerateSessionsW(WTS_CURRENT_SERVER_HANDLE) failed"; for (DWORD i = 0; i < sessionCount; i++) { WTS_SESSION_INFO_1W* session = &sessionInfo[i]; if (session->State != WTSActive) continue; FF_STRBUF_AUTO_DESTROY userName = ffStrbufCreateWS(session->pUserName); if (options->myselfOnly && !ffStrbufEqual(&instance.state.platform.userName, &userName)) continue; FFUserResult* user = (FFUserResult*) ffListAdd(users); ffStrbufInitMove(&user->name, &userName); ffStrbufInitWS(&user->hostName, session->pHostName); ffStrbufInitWS(&user->sessionName, session->pSessionName); ffStrbufInit(&user->clientIp); user->loginTime = 0; DWORD bytes = 0; PWTS_CLIENT_ADDRESS address = NULL; if (WTSQuerySessionInformationW(WTS_CURRENT_SERVER_HANDLE, session->SessionId, WTSClientAddress, (LPWSTR *) &address, &bytes)) { if (address->AddressFamily == AF_INET) ffStrbufSetF(&user->clientIp, "%u.%u.%u.%u", address->Address[2], address->Address[3], address->Address[4], address->Address[5]); else if (address->AddressFamily == AF_INET6) { char ipStr[INET6_ADDRSTRLEN]; if (inet_ntop(AF_INET6, address->Address, ipStr, sizeof(ipStr)) != NULL) ffStrbufSetS(&user->clientIp, ipStr); } WTSFreeMemory(address); } bytes = 0; PWTSINFOW wtsInfo = NULL; if (WTSQuerySessionInformationW(WTS_CURRENT_SERVER_HANDLE, session->SessionId, WTSSessionInfo, (LPWSTR *) &wtsInfo, &bytes)) { user->loginTime = to_ms(*(uint64_t*) &wtsInfo->LogonTime); WTSFreeMemory(wtsInfo); } } WTSFreeMemoryExW(WTSTypeSessionInfoLevel1, sessionInfo, 1); return NULL; }