2017-08-04 12:19:56 +02:00
|
|
|
|
/*
|
2024-03-05 18:13:14 +01:00
|
|
|
|
* Copyright (c) 2023 ETH Zürich, IT Services
|
2017-08-04 12:19:56 +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.Linq;
|
|
|
|
|
using System.Windows.Input;
|
2019-08-30 09:55:26 +02:00
|
|
|
|
using SafeExamBrowser.Logging.Contracts;
|
2019-09-05 09:00:41 +02:00
|
|
|
|
using SafeExamBrowser.Monitoring.Contracts.Keyboard;
|
2019-10-01 11:30:53 +02:00
|
|
|
|
using SafeExamBrowser.Settings.Monitoring;
|
|
|
|
|
using SafeExamBrowser.WindowsApi.Contracts;
|
|
|
|
|
using SafeExamBrowser.WindowsApi.Contracts.Events;
|
2017-08-04 12:19:56 +02:00
|
|
|
|
|
|
|
|
|
namespace SafeExamBrowser.Monitoring.Keyboard
|
|
|
|
|
{
|
|
|
|
|
public class KeyboardInterceptor : IKeyboardInterceptor
|
|
|
|
|
{
|
2019-10-01 11:30:53 +02:00
|
|
|
|
private Guid? hookId;
|
2023-07-21 09:31:59 +02:00
|
|
|
|
private readonly ILogger logger;
|
|
|
|
|
private readonly INativeMethods nativeMethods;
|
|
|
|
|
private readonly KeyboardSettings settings;
|
2017-08-04 12:19:56 +02:00
|
|
|
|
|
2019-10-01 16:24:10 +02:00
|
|
|
|
public KeyboardInterceptor(ILogger logger, INativeMethods nativeMethods, KeyboardSettings settings)
|
2017-08-04 12:19:56 +02:00
|
|
|
|
{
|
|
|
|
|
this.logger = logger;
|
2019-10-01 11:30:53 +02:00
|
|
|
|
this.nativeMethods = nativeMethods;
|
2017-08-04 12:19:56 +02:00
|
|
|
|
this.settings = settings;
|
|
|
|
|
}
|
|
|
|
|
|
2019-10-01 11:30:53 +02:00
|
|
|
|
public void Start()
|
|
|
|
|
{
|
|
|
|
|
hookId = nativeMethods.RegisterKeyboardHook(KeyboardHookCallback);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void Stop()
|
|
|
|
|
{
|
|
|
|
|
if (hookId.HasValue)
|
|
|
|
|
{
|
|
|
|
|
nativeMethods.DeregisterKeyboardHook(hookId.Value);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private bool KeyboardHookCallback(int keyCode, KeyModifier modifier, KeyState state)
|
2017-08-04 12:19:56 +02:00
|
|
|
|
{
|
|
|
|
|
var block = false;
|
|
|
|
|
var key = KeyInterop.KeyFromVirtualKey(keyCode);
|
|
|
|
|
|
|
|
|
|
block |= key == Key.Apps;
|
2019-01-09 16:01:56 +01:00
|
|
|
|
block |= key == Key.Escape && modifier == KeyModifier.None && !settings.AllowEsc;
|
|
|
|
|
block |= key == Key.F1 && !settings.AllowF1;
|
|
|
|
|
block |= key == Key.F2 && !settings.AllowF2;
|
|
|
|
|
block |= key == Key.F3 && !settings.AllowF3;
|
|
|
|
|
block |= key == Key.F4 && !settings.AllowF4;
|
|
|
|
|
block |= key == Key.F5 && !settings.AllowF5;
|
|
|
|
|
block |= key == Key.F6 && !settings.AllowF6;
|
|
|
|
|
block |= key == Key.F7 && !settings.AllowF7;
|
|
|
|
|
block |= key == Key.F8 && !settings.AllowF8;
|
|
|
|
|
block |= key == Key.F9 && !settings.AllowF9;
|
|
|
|
|
block |= key == Key.F10 && !settings.AllowF10;
|
|
|
|
|
block |= key == Key.F11 && !settings.AllowF11;
|
|
|
|
|
block |= key == Key.F12 && !settings.AllowF12;
|
|
|
|
|
block |= key == Key.LWin && !settings.AllowSystemKey;
|
|
|
|
|
block |= key == Key.PrintScreen && !settings.AllowPrintScreen;
|
|
|
|
|
block |= key == Key.RWin && !settings.AllowSystemKey;
|
2023-07-21 09:31:59 +02:00
|
|
|
|
|
2019-01-09 16:01:56 +01:00
|
|
|
|
block |= modifier.HasFlag(KeyModifier.Alt) && key == Key.Escape && !settings.AllowAltEsc;
|
|
|
|
|
block |= modifier.HasFlag(KeyModifier.Alt) && key == Key.F4 && !settings.AllowAltF4;
|
|
|
|
|
block |= modifier.HasFlag(KeyModifier.Alt) && key == Key.Space;
|
2019-11-21 08:45:38 +01:00
|
|
|
|
block |= modifier.HasFlag(KeyModifier.Alt) && key == Key.Tab;
|
2023-07-21 09:31:59 +02:00
|
|
|
|
|
|
|
|
|
block |= modifier.HasFlag(KeyModifier.Ctrl) && key == Key.C && !settings.AllowCtrlC;
|
2019-01-09 16:01:56 +01:00
|
|
|
|
block |= modifier.HasFlag(KeyModifier.Ctrl) && key == Key.Escape && !settings.AllowCtrlEsc;
|
2023-07-21 09:31:59 +02:00
|
|
|
|
block |= modifier.HasFlag(KeyModifier.Ctrl) && key == Key.V && !settings.AllowCtrlV;
|
|
|
|
|
block |= modifier.HasFlag(KeyModifier.Ctrl) && key == Key.X && !settings.AllowCtrlX;
|
2017-08-04 12:19:56 +02:00
|
|
|
|
|
|
|
|
|
if (block)
|
|
|
|
|
{
|
|
|
|
|
Log(key, keyCode, modifier, state);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return block;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void Log(Key key, int keyCode, KeyModifier modifier, KeyState state)
|
|
|
|
|
{
|
|
|
|
|
var modifierFlags = Enum.GetValues(typeof(KeyModifier)).OfType<KeyModifier>().Where(m => m != KeyModifier.None && modifier.HasFlag(m));
|
|
|
|
|
var modifiers = modifierFlags.Any() ? String.Join(" + ", modifierFlags) + " + " : string.Empty;
|
|
|
|
|
|
|
|
|
|
logger.Info($"Blocked '{modifiers}{key}' ({key} = {keyCode}) when {state.ToString().ToLower()}.");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|