2018-08-16 11:23:37 +02:00
|
|
|
|
/*
|
2024-03-05 18:13:14 +01:00
|
|
|
|
* Copyright (c) 2023 ETH Zürich, IT Services
|
2018-08-16 11:23:37 +02:00
|
|
|
|
*
|
|
|
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
using System;
|
|
|
|
|
using System.ComponentModel;
|
|
|
|
|
using System.Runtime.InteropServices;
|
2019-08-30 09:55:26 +02:00
|
|
|
|
using SafeExamBrowser.Logging.Contracts;
|
2018-08-16 11:23:37 +02:00
|
|
|
|
using SafeExamBrowser.WindowsApi.Constants;
|
2023-10-04 14:48:08 +02:00
|
|
|
|
using SafeExamBrowser.WindowsApi.Contracts;
|
2018-08-16 11:23:37 +02:00
|
|
|
|
|
2023-10-04 14:48:08 +02:00
|
|
|
|
namespace SafeExamBrowser.WindowsApi.Desktops
|
2018-08-16 11:23:37 +02:00
|
|
|
|
{
|
|
|
|
|
public class DesktopFactory : IDesktopFactory
|
|
|
|
|
{
|
2023-10-04 14:48:08 +02:00
|
|
|
|
private readonly ILogger logger;
|
|
|
|
|
private readonly Random random;
|
2018-08-16 11:23:37 +02:00
|
|
|
|
|
|
|
|
|
public DesktopFactory(ILogger logger)
|
|
|
|
|
{
|
|
|
|
|
this.logger = logger;
|
2023-10-04 14:48:08 +02:00
|
|
|
|
this.random = new Random();
|
2018-08-16 11:23:37 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public IDesktop CreateNew(string name)
|
|
|
|
|
{
|
|
|
|
|
logger.Debug($"Attempting to create new desktop '{name}'...");
|
|
|
|
|
|
2019-03-06 16:10:00 +01:00
|
|
|
|
var handle = User32.CreateDesktop(name, IntPtr.Zero, IntPtr.Zero, 0, (uint) AccessMask.GENERIC_ALL, IntPtr.Zero);
|
2018-08-16 11:23:37 +02:00
|
|
|
|
|
|
|
|
|
if (handle == IntPtr.Zero)
|
|
|
|
|
{
|
|
|
|
|
logger.Error($"Failed to create new desktop '{name}'!");
|
|
|
|
|
|
|
|
|
|
throw new Win32Exception(Marshal.GetLastWin32Error());
|
|
|
|
|
}
|
|
|
|
|
|
2018-08-17 14:48:50 +02:00
|
|
|
|
var desktop = new Desktop(handle, name);
|
2018-08-16 11:23:37 +02:00
|
|
|
|
|
2018-08-17 14:48:50 +02:00
|
|
|
|
logger.Debug($"Successfully created desktop {desktop}.");
|
|
|
|
|
|
|
|
|
|
return desktop;
|
2018-08-16 11:23:37 +02:00
|
|
|
|
}
|
|
|
|
|
|
2023-10-04 14:48:08 +02:00
|
|
|
|
public IDesktop CreateRandom()
|
|
|
|
|
{
|
|
|
|
|
logger.Debug($"Attempting to create random desktop...");
|
|
|
|
|
|
|
|
|
|
var name = GenerateRandomDesktopName();
|
|
|
|
|
var handle = User32.CreateDesktop(name, IntPtr.Zero, IntPtr.Zero, 0, (uint) AccessMask.GENERIC_ALL, IntPtr.Zero);
|
|
|
|
|
|
|
|
|
|
if (handle == IntPtr.Zero)
|
|
|
|
|
{
|
|
|
|
|
logger.Error($"Failed to create random desktop '{name}'!");
|
|
|
|
|
|
|
|
|
|
throw new Win32Exception(Marshal.GetLastWin32Error());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var obfuscatedHandle = new IntPtr(random.Next(100, 10000));
|
|
|
|
|
var obfuscatedName = GenerateRandomDesktopName();
|
|
|
|
|
var desktop = new ObfuscatedDesktop(handle, name, obfuscatedHandle, obfuscatedName);
|
|
|
|
|
|
|
|
|
|
logger.Debug($"Successfully created random desktop {desktop}.");
|
|
|
|
|
|
|
|
|
|
return desktop;
|
|
|
|
|
}
|
|
|
|
|
|
2018-08-16 11:23:37 +02:00
|
|
|
|
public IDesktop GetCurrent()
|
|
|
|
|
{
|
|
|
|
|
var threadId = Kernel32.GetCurrentThreadId();
|
|
|
|
|
var handle = User32.GetThreadDesktop(threadId);
|
2023-10-04 14:48:08 +02:00
|
|
|
|
var name = string.Empty;
|
2018-08-16 11:23:37 +02:00
|
|
|
|
var nameLength = 0;
|
|
|
|
|
|
2018-08-17 14:48:50 +02:00
|
|
|
|
if (handle == IntPtr.Zero)
|
2018-08-16 11:23:37 +02:00
|
|
|
|
{
|
2018-08-17 14:48:50 +02:00
|
|
|
|
logger.Error($"Failed to get desktop handle for thread with ID = {threadId}!");
|
2018-08-16 11:23:37 +02:00
|
|
|
|
|
|
|
|
|
throw new Win32Exception(Marshal.GetLastWin32Error());
|
|
|
|
|
}
|
|
|
|
|
|
2018-08-17 14:48:50 +02:00
|
|
|
|
logger.Debug($"Found desktop handle for thread with ID = {threadId}. Attempting to get desktop name...");
|
|
|
|
|
|
2018-08-16 11:23:37 +02:00
|
|
|
|
User32.GetUserObjectInformation(handle, Constant.UOI_NAME, IntPtr.Zero, 0, ref nameLength);
|
|
|
|
|
|
|
|
|
|
var namePointer = Marshal.AllocHGlobal(nameLength);
|
|
|
|
|
var success = User32.GetUserObjectInformation(handle, Constant.UOI_NAME, namePointer, nameLength, ref nameLength);
|
|
|
|
|
|
|
|
|
|
if (!success)
|
|
|
|
|
{
|
2018-08-17 14:48:50 +02:00
|
|
|
|
logger.Error($"Failed to retrieve name for desktop with handle = {handle}!");
|
2018-08-16 11:23:37 +02:00
|
|
|
|
|
|
|
|
|
throw new Win32Exception(Marshal.GetLastWin32Error());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
name = Marshal.PtrToStringAnsi(namePointer);
|
|
|
|
|
Marshal.FreeHGlobal(namePointer);
|
|
|
|
|
|
2018-08-17 14:48:50 +02:00
|
|
|
|
var desktop = new Desktop(handle, name);
|
|
|
|
|
|
|
|
|
|
logger.Debug($"Successfully determined current desktop {desktop}.");
|
2018-08-16 11:23:37 +02:00
|
|
|
|
|
2018-08-17 14:48:50 +02:00
|
|
|
|
return desktop;
|
2018-08-16 11:23:37 +02:00
|
|
|
|
}
|
2023-10-04 14:48:08 +02:00
|
|
|
|
|
|
|
|
|
private string GenerateRandomDesktopName()
|
|
|
|
|
{
|
|
|
|
|
var length = random.Next(5, 20);
|
|
|
|
|
var name = new char[length];
|
|
|
|
|
|
|
|
|
|
for (var letter = 0; letter < length; letter++)
|
|
|
|
|
{
|
|
|
|
|
name[letter] = (char) (random.Next(2) == 0 && letter != 0 ? random.Next('a', 'z' + 1) : random.Next('A', 'Z' + 1));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return new string(name);
|
|
|
|
|
}
|
2018-08-16 11:23:37 +02:00
|
|
|
|
}
|
|
|
|
|
}
|