Compare commits
212 commits
Author | SHA1 | Date | |
---|---|---|---|
|
7029b12d81 | ||
|
9762019499 | ||
|
d5b182ae2f | ||
|
4c0f3cfa6c | ||
|
f096b96741 | ||
|
50ac28f9ea | ||
|
144c3ba752 | ||
|
21353e6d6d | ||
|
26f14f235d | ||
|
1ff7d84375 | ||
|
febfd944e0 | ||
|
a1bfaadcd9 | ||
|
0b1746a82e | ||
|
ede6a926cc | ||
|
d4f5f203db | ||
|
a350949b1b | ||
|
6a77a41564 | ||
|
1f50ab74c9 | ||
|
b48ef21708 | ||
|
f3a9030505 | ||
|
89091acaac | ||
|
04843d3fa8 | ||
|
a3c9271faf | ||
|
68d6d47fe6 | ||
|
b4366adab1 | ||
|
c23b78488c | ||
|
58c8e69716 | ||
|
0fb7f23bcb | ||
|
05f46cd6b4 | ||
|
f2798581a4 | ||
|
471e69d460 | ||
|
62dc690a52 | ||
|
a41b40d428 | ||
|
8cf214b39c | ||
|
04dce13d86 | ||
|
767ac84391 | ||
|
84bbcb82ef | ||
|
b3228aedef | ||
|
3b099688f7 | ||
|
639700abd8 | ||
|
473edc7a2e | ||
|
60ee95a9ee | ||
|
1edde7b6f5 | ||
|
4015e9a574 | ||
|
bbb5ec2571 | ||
|
8b3f9b0838 | ||
|
e4a82e2f63 | ||
|
01db8fd84e | ||
|
a397446252 | ||
|
e8ebd2840e | ||
|
d9662ec31e | ||
|
c2f61ea6ab | ||
|
7801d68b97 | ||
|
ff16743ae7 | ||
|
832eee17d5 | ||
|
2c39668667 | ||
|
577a23b8b4 | ||
|
514414e322 | ||
|
4b222df6c5 | ||
|
acb9b97854 | ||
|
8a47228881 | ||
|
ff33394565 | ||
|
e6e0cca292 | ||
|
3f90d3a58a | ||
|
956771c0e7 | ||
|
9bbcaa2a98 | ||
|
a81837faa0 | ||
|
c46d1a3ade | ||
|
a1d62dd3de | ||
|
9045b852d0 | ||
|
ff5b91c010 | ||
|
787c84cc0e | ||
|
0777644f0e | ||
|
e5c02a1f74 | ||
|
91f2c14a77 | ||
|
4aacf85e9a | ||
|
a021bebde6 | ||
|
f902ee9598 | ||
|
ea9d7e0de7 | ||
|
a213ec0f7d | ||
|
731a748552 | ||
|
cb81906945 | ||
|
456894edb9 | ||
|
acb6b8cf09 | ||
|
6b40b64590 | ||
|
70ba9ad7b6 | ||
|
b62a8bdfe3 | ||
|
654aa14bee | ||
|
1925231a19 | ||
|
55c36f6d7d | ||
|
de5691cb25 | ||
|
93cee788f8 | ||
|
73fefad434 | ||
|
23de2bf8c7 | ||
|
96f67c2085 | ||
|
c9db98159d | ||
|
c52c461dbf | ||
|
cecfe095a7 | ||
|
5f6a57cd24 | ||
|
1e9d37ac13 | ||
|
ef267ef186 | ||
|
ecc8416dfb | ||
|
eae6ab1bdf | ||
|
27155a057d | ||
|
79dedf12b5 | ||
|
8c45af88fb | ||
|
181346b810 | ||
|
04571f51b2 | ||
|
ebca114c2e | ||
|
98fb7a32db | ||
|
1aa32403a7 | ||
|
4e152c26f1 | ||
|
945b9223e7 | ||
|
622df39fca | ||
|
5ef03d4101 | ||
|
bc8235951d | ||
|
ae352a883c | ||
|
7f4aee9058 | ||
|
c535124575 | ||
|
54b4444f8e | ||
|
51c21f8934 | ||
|
f8b354623a | ||
|
5a58a11dd0 | ||
|
e9976ac158 | ||
|
fb70c9c928 | ||
|
09750cefd0 | ||
|
a286b615f4 | ||
|
e563767d6e | ||
|
6f175bf0e7 | ||
|
cb47c985e9 | ||
|
af5e33c2d8 | ||
|
afe8b4bcca | ||
|
c6a8996138 | ||
|
bfd1da3a86 | ||
|
8e17504d35 | ||
|
499629d848 | ||
|
25cb91899c | ||
|
00a562b3c1 | ||
|
a3d0ab433b | ||
|
751bfcb144 | ||
|
8c3d9a31d7 | ||
|
75016158c5 | ||
|
4ac982a3dd | ||
|
ca02b1d674 | ||
|
400b259af7 | ||
|
421b3db53e | ||
|
2ef7c2c5ec | ||
|
f7479cd1a8 | ||
|
c44bac79fd | ||
|
026d1fbfd8 | ||
|
3711555f70 | ||
|
bbfa720b21 | ||
|
722d84978c | ||
|
fa16710bdb | ||
|
d76dbf6b40 | ||
|
b36df9ad5a | ||
|
bd993ecc6b | ||
|
fcebf4b436 | ||
|
c498ef9af1 | ||
|
0769cf6b4b | ||
|
56732537f8 | ||
|
27f2fde904 | ||
|
44432ab023 | ||
|
eff0051469 | ||
|
bd3b348f6a | ||
|
9b0bfa291e | ||
|
5173bf3d6e | ||
|
c1307624d9 | ||
|
210a0419ca | ||
|
f2917f69a6 | ||
|
bc30e56e38 | ||
|
a21c9007ab | ||
|
689e388e23 | ||
|
3b8f552138 | ||
|
e99bdabc51 | ||
|
7fc31f6e90 | ||
|
940baae655 | ||
|
8543c81867 | ||
|
817f598d8a | ||
|
37e3950a6f | ||
|
3dd023b285 | ||
|
543ad7040b | ||
|
5284a52278 | ||
|
204db744aa | ||
|
fd55367a7d | ||
|
11b10e8e45 | ||
|
627c568400 | ||
|
9507888900 | ||
|
82908607e5 | ||
|
23dd94d23c | ||
|
3b8c63ab56 | ||
|
40a7f63524 | ||
|
16fa6a0473 | ||
|
1605f0d00e | ||
|
e3c26335a9 | ||
|
989d414152 | ||
|
e33a12e7ec | ||
|
557e8a6be4 | ||
|
ba128bb6ac | ||
|
e4e0f7c16b | ||
|
efb3c8056a | ||
|
c201389af4 | ||
|
538127661f | ||
|
250ddb5bc9 | ||
|
da609f62c8 | ||
|
bf00b3883b | ||
|
390019048e | ||
|
22ef7ef364 | ||
|
1fec696909 | ||
|
bd145e14b0 | ||
|
c0f37b309b | ||
|
71b722d215 |
1096 changed files with 18871 additions and 5192 deletions
8
.github/ISSUE_TEMPLATE/bug-report.md
vendored
8
.github/ISSUE_TEMPLATE/bug-report.md
vendored
|
@ -1,15 +1,15 @@
|
||||||
---
|
---
|
||||||
name: Bug Report
|
name: Bug Report
|
||||||
about: Create a report to help us improve Safe Exam Browser
|
about: Create a bug report to help us improve Safe Exam Browser.
|
||||||
title: ''
|
title: ''
|
||||||
labels: ''
|
labels: ''
|
||||||
assignees: dbuechel
|
assignees: dbuechel
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
**IMPORTANT**
|
> [!IMPORTANT]
|
||||||
Please _always_ consult the documentation first before creating a bug report!
|
> - Please _always_ consult the documentation first before creating a bug report: https://safeexambrowser.org/windows/win_usermanual_en.html.
|
||||||
https://safeexambrowser.org/windows/win_usermanual_en.html
|
> - Please _always_ attach the log file(s) of the affected session(s)! They can be found under `%LocalAppData%\SafeExamBrowser\Logs`.
|
||||||
|
|
||||||
**Describe the Bug**
|
**Describe the Bug**
|
||||||
A clear and concise description of what the bug is.
|
A clear and concise description of what the bug is.
|
||||||
|
|
2
.github/ISSUE_TEMPLATE/feature-request.md
vendored
2
.github/ISSUE_TEMPLATE/feature-request.md
vendored
|
@ -1,6 +1,6 @@
|
||||||
---
|
---
|
||||||
name: Feature Request
|
name: Feature Request
|
||||||
about: Suggest an idea for Safe Exam Browser
|
about: Suggest an idea or new feature for Safe Exam Browser.
|
||||||
title: ''
|
title: ''
|
||||||
labels: ''
|
labels: ''
|
||||||
assignees: dbuechel
|
assignees: dbuechel
|
||||||
|
|
54
.github/workflows/codeql.yml
vendored
Normal file
54
.github/workflows/codeql.yml
vendored
Normal file
|
@ -0,0 +1,54 @@
|
||||||
|
name: "CodeQL"
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches: [ "master", "*" ]
|
||||||
|
pull_request:
|
||||||
|
branches: [ "master", "*" ]
|
||||||
|
schedule:
|
||||||
|
- cron: '0 0 * * 1'
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
analyze:
|
||||||
|
name: Analyze
|
||||||
|
runs-on: "windows-latest"
|
||||||
|
timeout-minutes: 360
|
||||||
|
permissions:
|
||||||
|
actions: read
|
||||||
|
contents: read
|
||||||
|
security-events: write
|
||||||
|
|
||||||
|
strategy:
|
||||||
|
fail-fast: false
|
||||||
|
matrix:
|
||||||
|
language: [ 'csharp', 'javascript-typescript' ]
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout repository
|
||||||
|
uses: actions/checkout@v3
|
||||||
|
|
||||||
|
- name: Initialize CodeQL
|
||||||
|
uses: github/codeql-action/init@v2
|
||||||
|
with:
|
||||||
|
languages: ${{ matrix.language }}
|
||||||
|
queries: security-extended,security-and-quality
|
||||||
|
|
||||||
|
# Autobuild attempts to build any compiled languages (C/C++, C#, Go, Java, or Swift).
|
||||||
|
# If this step fails, then you should remove it and run the build manually (see below)
|
||||||
|
- name: Autobuild
|
||||||
|
uses: github/codeql-action/autobuild@v2
|
||||||
|
|
||||||
|
# ℹ️ Command-line programs to run using the OS shell.
|
||||||
|
# 📚 See https://docs.github.com/en/actions/using-workflows/workflow-syntax-for-github-actions#jobsjob_idstepsrun
|
||||||
|
|
||||||
|
# If the Autobuild fails above, remove it and uncomment the following three lines.
|
||||||
|
# modify them (or add more) to build your code if your project, please refer to the EXAMPLE below for guidance.
|
||||||
|
|
||||||
|
# - run: |
|
||||||
|
# echo "Run, Build Application using script"
|
||||||
|
# ./location_of_script_within_repo/buildscript.sh
|
||||||
|
|
||||||
|
- name: Perform CodeQL Analysis
|
||||||
|
uses: github/codeql-action/analyze@v2
|
||||||
|
with:
|
||||||
|
category: "/language:${{matrix.language}}"
|
25
.github/workflows/issues.yml
vendored
Normal file
25
.github/workflows/issues.yml
vendored
Normal file
|
@ -0,0 +1,25 @@
|
||||||
|
name: Issue Maintenance
|
||||||
|
on:
|
||||||
|
workflow_dispatch:
|
||||||
|
schedule:
|
||||||
|
- cron: "0 0 * * *"
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
close-issues:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
permissions:
|
||||||
|
issues: write
|
||||||
|
pull-requests: write
|
||||||
|
steps:
|
||||||
|
- uses: actions/stale@v8
|
||||||
|
with:
|
||||||
|
# https://github.com/marketplace/actions/close-stale-issues
|
||||||
|
days-before-issue-stale: 28
|
||||||
|
days-before-issue-close: 14
|
||||||
|
stale-issue-label: "stale"
|
||||||
|
stale-issue-message: "This issue is stale because it has been open for 28 days with no activity. It will soon be closed automatically if there are no updates."
|
||||||
|
close-issue-message: "This issue was closed because it has been inactive for 14 days since being marked as stale."
|
||||||
|
days-before-pr-stale: -1
|
||||||
|
days-before-pr-close: -1
|
||||||
|
exempt-issue-labels: "bug,enhancement,feature request,known issue"
|
||||||
|
repo-token: ${{ secrets.GITHUB_TOKEN }}
|
Binary file not shown.
|
@ -6,18 +6,17 @@ Refactored version of Safe Exam Browser for Windows with Chromium as integrated
|
||||||
|
|
||||||
SEB 3.x requires the prerequisites listed below in order to work correctly. These are automatically installed with the setup bundle and need only be manually installed when using the MSI packages.
|
SEB 3.x requires the prerequisites listed below in order to work correctly. These are automatically installed with the setup bundle and need only be manually installed when using the MSI packages.
|
||||||
|
|
||||||
* .NET Framework 4.7.2 Runtime: https://dotnet.microsoft.com/download/dotnet-framework/net472
|
* .NET Framework 4.8 Runtime: https://dotnet.microsoft.com/download/dotnet-framework/net48
|
||||||
* Microsoft Edge WebView2 Runtime: https://go.microsoft.com/fwlink/p/?LinkId=2124703
|
|
||||||
* Visual C++ 2015-2019 Redistributable: https://support.microsoft.com/en-us/help/2977003/the-latest-supported-visual-c-downloads
|
* Visual C++ 2015-2019 Redistributable: https://support.microsoft.com/en-us/help/2977003/the-latest-supported-visual-c-downloads
|
||||||
|
|
||||||
## Project Status
|
## Project Status
|
||||||
|
|
||||||
**_DISCLAIMER_**\
|
> [!WARNING]
|
||||||
**The builds linked below are for testing purposes only.** They may be unstable and should thus _never_ be used in a production environment! Always use the latest, official release version of SEB.
|
> **The builds linked below are for testing purposes only.** They may be unstable and should thus _never_ be used in a production environment! Always use the latest, official release version of SEB.
|
||||||
|
|
||||||
| Aspect | Status | Details |
|
| Aspect | Status | Details |
|
||||||
| ----------------- | --------------------------------------------------------------------------------------------------------------------- | --------------------------------------------------------------- |
|
| ----------------- | --------------------------------------------------------------------------------------------------------------------- | --------------------------------------------------------------- |
|
||||||
| Development Build | ![Development Build Status](https://sebdev-let.ethz.ch/api/projects/status/kq78qrjtnpk82ti0?svg=true) | https://sebdev-let.ethz.ch/project/appveyor/seb-win-refactoring |
|
| Development Build | ![Development Build Status](https://sebdev.ethz.ch/api/projects/status/kq78qrjtnpk82ti0?svg=true) | https://sebdev.ethz.ch/project/appveyor/seb-win-refactoring |
|
||||||
| Test Build | ![Test Build Status](https://ci.appveyor.com/api/projects/status/a56akt9r174570m7?svg=true) | https://ci.appveyor.com/project/dbuechel/seb-win-refactoring |
|
| Test Build | ![Test Build Status](https://ci.appveyor.com/api/projects/status/a56akt9r174570m7?svg=true) | https://ci.appveyor.com/project/dbuechel/seb-win-refactoring |
|
||||||
| Test Run | ![AppVeyor Tests](https://img.shields.io/appveyor/tests/dbuechel/seb-win-refactoring?logo=appveyor&logoColor=%23ccc) | https://ci.appveyor.com/project/dbuechel/seb-win-refactoring |
|
| Test Run | ![AppVeyor Tests](https://img.shields.io/appveyor/tests/dbuechel/seb-win-refactoring?logo=appveyor&logoColor=%23ccc) | https://ci.appveyor.com/project/dbuechel/seb-win-refactoring |
|
||||||
| Code Coverage | ![Code Coverage](https://codecov.io/gh/SafeExamBrowser/seb-win-refactoring/branch/master/graph/badge.svg) | https://codecov.io/gh/SafeExamBrowser/seb-win-refactoring |
|
| Code Coverage | ![Code Coverage](https://codecov.io/gh/SafeExamBrowser/seb-win-refactoring/branch/master/graph/badge.svg) | https://codecov.io/gh/SafeExamBrowser/seb-win-refactoring |
|
||||||
|
|
35
SECURITY.md
Normal file
35
SECURITY.md
Normal file
|
@ -0,0 +1,35 @@
|
||||||
|
# Security Policy
|
||||||
|
|
||||||
|
We only support the latest official relese version with respect to security vulnerabilities. Thus, only the latest or then the upcoming next release version
|
||||||
|
will receive vulnerability fixes and security updates. A vulnerability may however be reported for any version, unless it already has been fixed with a later
|
||||||
|
release version.
|
||||||
|
|
||||||
|
## Reporting a Vulnerability
|
||||||
|
|
||||||
|
> [!IMPORTANT]
|
||||||
|
> - Please _always_ verify that no later release version exists which fixes the vulnerability.
|
||||||
|
> - Please _always_ consult the documentation first before creating a vulnerability report: https://safeexambrowser.org/windows/win_usermanual_en.html.
|
||||||
|
> - Please _always_ attach the log file(s) of the affected session(s)! They can be found under `%LocalAppData%\SafeExamBrowser\Logs`.
|
||||||
|
|
||||||
|
**Describe the Vulnerability**
|
||||||
|
A clear and concise description of what the vulnerability is.
|
||||||
|
|
||||||
|
**Steps to Reproduce**
|
||||||
|
Steps to reproduce the behavior:
|
||||||
|
1. Go to '...'
|
||||||
|
2. Click on '....'
|
||||||
|
3. Scroll down to '....'
|
||||||
|
4. See ...
|
||||||
|
|
||||||
|
**Expected Behavior**
|
||||||
|
A clear and concise description of what you expected to happen.
|
||||||
|
|
||||||
|
**Screenshots**
|
||||||
|
If applicable, add screenshots to help explain your problem.
|
||||||
|
|
||||||
|
**Version Information**
|
||||||
|
- OS: [e.g. Windows 10 Professional, Version 1803]
|
||||||
|
- SEB-Version [e.g. SEB 3.0.1]
|
||||||
|
|
||||||
|
**Additional Context**
|
||||||
|
Add any other context about the vulnerability here.
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
@ -16,7 +16,7 @@ namespace SafeExamBrowser.Applications.Contracts
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Controls the lifetime and functionality of an application.
|
/// Controls the lifetime and functionality of an application.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public interface IApplication
|
public interface IApplication<out TWindow> where TWindow : IApplicationWindow
|
||||||
{
|
{
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Indicates whether the application should be automatically started.
|
/// Indicates whether the application should be automatically started.
|
||||||
|
@ -51,7 +51,7 @@ namespace SafeExamBrowser.Applications.Contracts
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Returns all windows of the application.
|
/// Returns all windows of the application.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
IEnumerable<IApplicationWindow> GetWindows();
|
IEnumerable<TWindow> GetWindows();
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Performs any initialization work, if necessary.
|
/// Performs any initialization work, if necessary.
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
@ -18,6 +18,6 @@ namespace SafeExamBrowser.Applications.Contracts
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Attempts to create an application according to the given settings.
|
/// Attempts to create an application according to the given settings.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
FactoryResult TryCreate(WhitelistApplication settings, out IApplication application);
|
FactoryResult TryCreate(WhitelistApplication settings, out IApplication<IApplicationWindow> application);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
@ -13,7 +13,7 @@ using SafeExamBrowser.Core.Contracts.Resources.Icons;
|
||||||
namespace SafeExamBrowser.Applications.Contracts
|
namespace SafeExamBrowser.Applications.Contracts
|
||||||
{
|
{
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Defines a window of an <see cref="IApplication"/>.
|
/// Defines a window of an <see cref="IApplication{TWindow}"/>.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public interface IApplicationWindow
|
public interface IApplicationWindow
|
||||||
{
|
{
|
||||||
|
|
|
@ -8,7 +8,7 @@ using System.Runtime.InteropServices;
|
||||||
[assembly: AssemblyDescription("Safe Exam Browser")]
|
[assembly: AssemblyDescription("Safe Exam Browser")]
|
||||||
[assembly: AssemblyCompany("ETH Zürich")]
|
[assembly: AssemblyCompany("ETH Zürich")]
|
||||||
[assembly: AssemblyProduct("SafeExamBrowser.Applications.Contracts")]
|
[assembly: AssemblyProduct("SafeExamBrowser.Applications.Contracts")]
|
||||||
[assembly: AssemblyCopyright("Copyright © 2023 ETH Zürich, Educational Development and Technology (LET)")]
|
[assembly: AssemblyCopyright("Copyright © 2024 ETH Zürich, IT Services")]
|
||||||
|
|
||||||
// Setting ComVisible to false makes the types in this assembly not visible
|
// Setting ComVisible to false makes the types in this assembly not visible
|
||||||
// to COM components. If you need to access a type in this assembly from
|
// to COM components. If you need to access a type in this assembly from
|
||||||
|
|
|
@ -9,9 +9,10 @@
|
||||||
<AppDesignerFolder>Properties</AppDesignerFolder>
|
<AppDesignerFolder>Properties</AppDesignerFolder>
|
||||||
<RootNamespace>SafeExamBrowser.Applications.Contracts</RootNamespace>
|
<RootNamespace>SafeExamBrowser.Applications.Contracts</RootNamespace>
|
||||||
<AssemblyName>SafeExamBrowser.Applications.Contracts</AssemblyName>
|
<AssemblyName>SafeExamBrowser.Applications.Contracts</AssemblyName>
|
||||||
<TargetFrameworkVersion>v4.7.2</TargetFrameworkVersion>
|
<TargetFrameworkVersion>v4.8</TargetFrameworkVersion>
|
||||||
<FileAlignment>512</FileAlignment>
|
<FileAlignment>512</FileAlignment>
|
||||||
<Deterministic>true</Deterministic>
|
<Deterministic>true</Deterministic>
|
||||||
|
<TargetFrameworkProfile />
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
<PropertyGroup Condition="'$(Configuration)|$(Platform)' == 'Debug|x86'">
|
<PropertyGroup Condition="'$(Configuration)|$(Platform)' == 'Debug|x86'">
|
||||||
<DebugSymbols>true</DebugSymbols>
|
<DebugSymbols>true</DebugSymbols>
|
||||||
|
|
|
@ -0,0 +1,126 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
|
*
|
||||||
|
* 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 Microsoft.VisualStudio.TestTools.UnitTesting;
|
||||||
|
using Moq;
|
||||||
|
using SafeExamBrowser.Applications.Contracts;
|
||||||
|
using SafeExamBrowser.Logging.Contracts;
|
||||||
|
using SafeExamBrowser.Monitoring.Contracts.Applications;
|
||||||
|
using SafeExamBrowser.Settings.Applications;
|
||||||
|
using SafeExamBrowser.SystemComponents.Contracts.Registry;
|
||||||
|
using SafeExamBrowser.WindowsApi.Contracts;
|
||||||
|
|
||||||
|
namespace SafeExamBrowser.Applications.UnitTests
|
||||||
|
{
|
||||||
|
[TestClass]
|
||||||
|
public class ApplicationFactoryTests
|
||||||
|
{
|
||||||
|
private Mock<IApplicationMonitor> applicationMonitor;
|
||||||
|
private Mock<IModuleLogger> logger;
|
||||||
|
private Mock<INativeMethods> nativeMethods;
|
||||||
|
private Mock<IProcessFactory> processFactory;
|
||||||
|
private Mock<IRegistry> registry;
|
||||||
|
|
||||||
|
private ApplicationFactory sut;
|
||||||
|
|
||||||
|
[TestInitialize]
|
||||||
|
public void Initialize()
|
||||||
|
{
|
||||||
|
applicationMonitor = new Mock<IApplicationMonitor>();
|
||||||
|
logger = new Mock<IModuleLogger>();
|
||||||
|
nativeMethods = new Mock<INativeMethods>();
|
||||||
|
processFactory = new Mock<IProcessFactory>();
|
||||||
|
registry = new Mock<IRegistry>();
|
||||||
|
|
||||||
|
sut = new ApplicationFactory(applicationMonitor.Object, logger.Object, nativeMethods.Object, processFactory.Object, registry.Object);
|
||||||
|
}
|
||||||
|
|
||||||
|
[TestMethod]
|
||||||
|
public void MustCorrectlyCreateApplication()
|
||||||
|
{
|
||||||
|
var settings = new WhitelistApplication
|
||||||
|
{
|
||||||
|
DisplayName = "Windows Command Prompt",
|
||||||
|
ExecutableName = "cmd.exe",
|
||||||
|
};
|
||||||
|
|
||||||
|
var result = sut.TryCreate(settings, out var application);
|
||||||
|
|
||||||
|
Assert.AreEqual(FactoryResult.Success, result);
|
||||||
|
Assert.IsNotNull(application);
|
||||||
|
Assert.IsInstanceOfType<ExternalApplication>(application);
|
||||||
|
}
|
||||||
|
|
||||||
|
[TestMethod]
|
||||||
|
public void MustCorrectlyReadPathFromRegistry()
|
||||||
|
{
|
||||||
|
object o = @"C:\Some\Registry\Path";
|
||||||
|
var settings = new WhitelistApplication
|
||||||
|
{
|
||||||
|
DisplayName = "Windows Command Prompt",
|
||||||
|
ExecutableName = "cmd.exe",
|
||||||
|
ExecutablePath = @"C:\Some\Path"
|
||||||
|
};
|
||||||
|
|
||||||
|
registry.Setup(r => r.TryRead(It.Is<string>(s => s.Contains(RegistryValue.MachineHive.AppPaths_Key)), It.Is<string>(s => s == "Path"), out o)).Returns(true);
|
||||||
|
|
||||||
|
var result = sut.TryCreate(settings, out var application);
|
||||||
|
|
||||||
|
registry.Verify(r => r.TryRead(It.Is<string>(s => s.Contains(RegistryValue.MachineHive.AppPaths_Key)), It.Is<string>(s => s == "Path"), out o), Times.Once);
|
||||||
|
|
||||||
|
Assert.AreEqual(FactoryResult.Success, result);
|
||||||
|
Assert.IsNotNull(application);
|
||||||
|
Assert.IsInstanceOfType<ExternalApplication>(application);
|
||||||
|
}
|
||||||
|
|
||||||
|
[TestMethod]
|
||||||
|
public void MustIndicateIfApplicationNotFound()
|
||||||
|
{
|
||||||
|
var settings = new WhitelistApplication
|
||||||
|
{
|
||||||
|
ExecutableName = "some_random_application_which_does_not_exist_on_a_normal_system.exe",
|
||||||
|
ExecutablePath = "Some/Path/Which/Does/Not/Exist"
|
||||||
|
};
|
||||||
|
|
||||||
|
var result = sut.TryCreate(settings, out var application);
|
||||||
|
|
||||||
|
Assert.AreEqual(FactoryResult.NotFound, result);
|
||||||
|
Assert.IsNull(application);
|
||||||
|
}
|
||||||
|
|
||||||
|
[TestMethod]
|
||||||
|
public void MustFailGracefullyWhenPathIsInvalid()
|
||||||
|
{
|
||||||
|
var settings = new WhitelistApplication
|
||||||
|
{
|
||||||
|
ExecutableName = "asdfg(/ç)&=%\"fsdg..exe..",
|
||||||
|
ExecutablePath = "[]#°§¬#°¢@tu03450'w89tz!$£äöüèé:"
|
||||||
|
};
|
||||||
|
|
||||||
|
var result = sut.TryCreate(settings, out _);
|
||||||
|
|
||||||
|
logger.Verify(l => l.Error(It.IsAny<string>(), It.IsAny<Exception>()), Times.AtLeastOnce);
|
||||||
|
|
||||||
|
Assert.AreEqual(FactoryResult.NotFound, result);
|
||||||
|
}
|
||||||
|
|
||||||
|
[TestMethod]
|
||||||
|
public void MustFailGracefullyAndIndicateThatErrorOccurred()
|
||||||
|
{
|
||||||
|
var o = default(object);
|
||||||
|
var settings = new WhitelistApplication();
|
||||||
|
|
||||||
|
registry.Setup(r => r.TryRead(It.IsAny<string>(), It.IsAny<string>(), out o)).Throws<Exception>();
|
||||||
|
|
||||||
|
var result = sut.TryCreate(settings, out var application);
|
||||||
|
|
||||||
|
Assert.AreEqual(FactoryResult.Error, result);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,62 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
|
*
|
||||||
|
* 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 Microsoft.VisualStudio.TestTools.UnitTesting;
|
||||||
|
using Moq;
|
||||||
|
using SafeExamBrowser.Core.Contracts.Resources.Icons;
|
||||||
|
using SafeExamBrowser.Logging.Contracts;
|
||||||
|
using SafeExamBrowser.WindowsApi.Contracts;
|
||||||
|
|
||||||
|
namespace SafeExamBrowser.Applications.UnitTests
|
||||||
|
{
|
||||||
|
[TestClass]
|
||||||
|
public class ExternalApplicationInstanceTests
|
||||||
|
{
|
||||||
|
private NativeIconResource icon;
|
||||||
|
private Mock<ILogger> logger;
|
||||||
|
private Mock<INativeMethods> nativeMethods;
|
||||||
|
private Mock<IProcess> process;
|
||||||
|
private ExternalApplicationInstance sut;
|
||||||
|
|
||||||
|
[TestInitialize]
|
||||||
|
public void Initialize()
|
||||||
|
{
|
||||||
|
icon = new NativeIconResource();
|
||||||
|
logger = new Mock<ILogger>();
|
||||||
|
nativeMethods = new Mock<INativeMethods>();
|
||||||
|
process = new Mock<IProcess>();
|
||||||
|
|
||||||
|
sut = new ExternalApplicationInstance(icon, logger.Object, nativeMethods.Object, process.Object, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
[TestMethod]
|
||||||
|
public void Terminate_MustDoNothingIfAlreadyTerminated()
|
||||||
|
{
|
||||||
|
process.SetupGet(p => p.HasTerminated).Returns(true);
|
||||||
|
|
||||||
|
sut.Terminate();
|
||||||
|
|
||||||
|
process.Verify(p => p.TryClose(It.IsAny<int>()), Times.Never());
|
||||||
|
process.Verify(p => p.TryKill(It.IsAny<int>()), Times.Never());
|
||||||
|
}
|
||||||
|
|
||||||
|
[TestMethod]
|
||||||
|
public void Terminate_MustLogIfTerminationFailed()
|
||||||
|
{
|
||||||
|
process.Setup(p => p.TryClose(It.IsAny<int>())).Returns(false);
|
||||||
|
process.Setup(p => p.TryKill(It.IsAny<int>())).Returns(false);
|
||||||
|
process.SetupGet(p => p.HasTerminated).Returns(false);
|
||||||
|
|
||||||
|
sut.Terminate();
|
||||||
|
|
||||||
|
logger.Verify(l => l.Warn(It.IsAny<string>()), Times.AtLeastOnce);
|
||||||
|
process.Verify(p => p.TryClose(It.IsAny<int>()), Times.AtLeastOnce());
|
||||||
|
process.Verify(p => p.TryKill(It.IsAny<int>()), Times.AtLeastOnce());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,216 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
|
*
|
||||||
|
* 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.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Threading;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using Microsoft.VisualStudio.TestTools.UnitTesting;
|
||||||
|
using Moq;
|
||||||
|
using SafeExamBrowser.Core.Contracts.Resources.Icons;
|
||||||
|
using SafeExamBrowser.Logging.Contracts;
|
||||||
|
using SafeExamBrowser.Monitoring.Contracts.Applications;
|
||||||
|
using SafeExamBrowser.Monitoring.Contracts.Applications.Events;
|
||||||
|
using SafeExamBrowser.Settings.Applications;
|
||||||
|
using SafeExamBrowser.WindowsApi.Contracts;
|
||||||
|
|
||||||
|
namespace SafeExamBrowser.Applications.UnitTests
|
||||||
|
{
|
||||||
|
[TestClass]
|
||||||
|
public class ExternalApplicationTests
|
||||||
|
{
|
||||||
|
private Mock<IApplicationMonitor> applicationMonitor;
|
||||||
|
private string executablePath;
|
||||||
|
private Mock<IModuleLogger> logger;
|
||||||
|
private Mock<INativeMethods> nativeMethods;
|
||||||
|
private Mock<IProcessFactory> processFactory;
|
||||||
|
private WhitelistApplication settings;
|
||||||
|
|
||||||
|
private ExternalApplication sut;
|
||||||
|
|
||||||
|
[TestInitialize]
|
||||||
|
public void Initialize()
|
||||||
|
{
|
||||||
|
applicationMonitor = new Mock<IApplicationMonitor>();
|
||||||
|
executablePath = @"C:\Some\Random\Path\Application.exe";
|
||||||
|
logger = new Mock<IModuleLogger>();
|
||||||
|
nativeMethods = new Mock<INativeMethods>();
|
||||||
|
processFactory = new Mock<IProcessFactory>();
|
||||||
|
settings = new WhitelistApplication();
|
||||||
|
|
||||||
|
logger.Setup(l => l.CloneFor(It.IsAny<string>())).Returns(new Mock<IModuleLogger>().Object);
|
||||||
|
|
||||||
|
sut = new ExternalApplication(applicationMonitor.Object, executablePath, logger.Object, nativeMethods.Object, processFactory.Object, settings, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
[TestMethod]
|
||||||
|
public void GetWindows_MustCorrectlyReturnOpenWindows()
|
||||||
|
{
|
||||||
|
var openWindows = new List<IntPtr> { new IntPtr(123), new IntPtr(234), new IntPtr(456), new IntPtr(345), new IntPtr(567), new IntPtr(789) };
|
||||||
|
var process1 = new Mock<IProcess>();
|
||||||
|
var process2 = new Mock<IProcess>();
|
||||||
|
var sync = new AutoResetEvent(false);
|
||||||
|
|
||||||
|
nativeMethods.Setup(n => n.GetOpenWindows()).Returns(openWindows);
|
||||||
|
nativeMethods.Setup(n => n.GetProcessIdFor(It.Is<IntPtr>(p => p == new IntPtr(234)))).Returns(1234);
|
||||||
|
nativeMethods.Setup(n => n.GetProcessIdFor(It.Is<IntPtr>(p => p == new IntPtr(345)))).Returns(1234);
|
||||||
|
nativeMethods.Setup(n => n.GetProcessIdFor(It.Is<IntPtr>(p => p == new IntPtr(567)))).Returns(5678);
|
||||||
|
process1.Setup(p => p.TryClose(It.IsAny<int>())).Returns(false);
|
||||||
|
process1.Setup(p => p.TryKill(It.IsAny<int>())).Returns(true);
|
||||||
|
process1.SetupGet(p => p.Id).Returns(1234);
|
||||||
|
process2.Setup(p => p.TryClose(It.IsAny<int>())).Returns(true);
|
||||||
|
process2.SetupGet(p => p.Id).Returns(5678);
|
||||||
|
processFactory.Setup(f => f.StartNew(It.IsAny<string>(), It.IsAny<string[]>())).Returns(process1.Object);
|
||||||
|
|
||||||
|
sut.WindowsChanged += () => sync.Set();
|
||||||
|
sut.Initialize();
|
||||||
|
sut.Start();
|
||||||
|
|
||||||
|
applicationMonitor.Raise(m => m.InstanceStarted += null, sut.Id, process2.Object);
|
||||||
|
|
||||||
|
sync.WaitOne();
|
||||||
|
sync.WaitOne();
|
||||||
|
|
||||||
|
var windows = sut.GetWindows();
|
||||||
|
|
||||||
|
Assert.AreEqual(3, windows.Count());
|
||||||
|
Assert.IsTrue(windows.Any(w => w.Handle == new IntPtr(234)));
|
||||||
|
Assert.IsTrue(windows.Any(w => w.Handle == new IntPtr(345)));
|
||||||
|
Assert.IsTrue(windows.Any(w => w.Handle == new IntPtr(567)));
|
||||||
|
|
||||||
|
nativeMethods.Setup(n => n.GetOpenWindows()).Returns(openWindows.Skip(2));
|
||||||
|
Task.Run(() => process2.Raise(p => p.Terminated += null, default(int)));
|
||||||
|
|
||||||
|
sync.WaitOne();
|
||||||
|
sync.WaitOne();
|
||||||
|
|
||||||
|
windows = sut.GetWindows();
|
||||||
|
|
||||||
|
Assert.AreEqual(1, windows.Count());
|
||||||
|
Assert.IsTrue(windows.Any(w => w.Handle != new IntPtr(234)));
|
||||||
|
Assert.IsTrue(windows.Any(w => w.Handle == new IntPtr(345)));
|
||||||
|
Assert.IsTrue(windows.All(w => w.Handle != new IntPtr(567)));
|
||||||
|
}
|
||||||
|
|
||||||
|
[TestMethod]
|
||||||
|
public void Initialize_MustInitializeCorrectly()
|
||||||
|
{
|
||||||
|
settings.AutoStart = new Random().Next(2) == 1;
|
||||||
|
settings.Description = "Some Description";
|
||||||
|
|
||||||
|
sut.Initialize();
|
||||||
|
|
||||||
|
applicationMonitor.VerifyAdd(a => a.InstanceStarted += It.IsAny<InstanceStartedEventHandler>(), Times.Once);
|
||||||
|
|
||||||
|
Assert.AreEqual(settings.AutoStart, sut.AutoStart);
|
||||||
|
Assert.AreEqual(executablePath, (sut.Icon as EmbeddedIconResource).FilePath);
|
||||||
|
Assert.AreEqual(settings.Id, settings.Id);
|
||||||
|
Assert.AreEqual(settings.DisplayName, sut.Name);
|
||||||
|
Assert.AreEqual(settings.Description ?? settings.DisplayName, sut.Tooltip);
|
||||||
|
}
|
||||||
|
|
||||||
|
[TestMethod]
|
||||||
|
public void Start_MustCreateInstanceCorrectly()
|
||||||
|
{
|
||||||
|
settings.Arguments.Add("some_parameter");
|
||||||
|
settings.Arguments.Add("another_parameter");
|
||||||
|
settings.Arguments.Add("yet another parameter");
|
||||||
|
|
||||||
|
sut.Start();
|
||||||
|
|
||||||
|
processFactory.Verify(f => f.StartNew(executablePath, It.Is<string[]>(args => args.All(a => settings.Arguments.Contains(a)))), Times.Once);
|
||||||
|
}
|
||||||
|
|
||||||
|
[TestMethod]
|
||||||
|
public void Start_MustHandleFailureGracefully()
|
||||||
|
{
|
||||||
|
processFactory.Setup(f => f.StartNew(It.IsAny<string>(), It.IsAny<string[]>())).Throws<Exception>();
|
||||||
|
|
||||||
|
sut.Start();
|
||||||
|
|
||||||
|
logger.Verify(l => l.Error(It.IsAny<string>(), It.IsAny<Exception>()), Times.AtLeastOnce);
|
||||||
|
processFactory.Verify(f => f.StartNew(It.IsAny<string>(), It.IsAny<string[]>()), Times.Once);
|
||||||
|
}
|
||||||
|
|
||||||
|
[TestMethod]
|
||||||
|
public void Start_MustRemoveInstanceCorrectlyWhenTerminated()
|
||||||
|
{
|
||||||
|
var eventCount = 0;
|
||||||
|
var openWindows = new List<IntPtr> { new IntPtr(123), new IntPtr(234), new IntPtr(456), new IntPtr(345), new IntPtr(567), new IntPtr(789), };
|
||||||
|
var process = new Mock<IProcess>();
|
||||||
|
var sync = new AutoResetEvent(false);
|
||||||
|
|
||||||
|
nativeMethods.Setup(n => n.GetOpenWindows()).Returns(openWindows);
|
||||||
|
nativeMethods.Setup(n => n.GetProcessIdFor(It.Is<IntPtr>(p => p == new IntPtr(234)))).Returns(1234);
|
||||||
|
process.Setup(p => p.TryClose(It.IsAny<int>())).Returns(false);
|
||||||
|
process.Setup(p => p.TryKill(It.IsAny<int>())).Returns(true);
|
||||||
|
process.SetupGet(p => p.Id).Returns(1234);
|
||||||
|
processFactory.Setup(f => f.StartNew(It.IsAny<string>(), It.IsAny<string[]>())).Returns(process.Object);
|
||||||
|
|
||||||
|
sut.WindowsChanged += () =>
|
||||||
|
{
|
||||||
|
eventCount++;
|
||||||
|
sync.Set();
|
||||||
|
};
|
||||||
|
|
||||||
|
sut.Initialize();
|
||||||
|
sut.Start();
|
||||||
|
|
||||||
|
sync.WaitOne();
|
||||||
|
|
||||||
|
Assert.AreEqual(1, sut.GetWindows().Count());
|
||||||
|
|
||||||
|
process.Raise(p => p.Terminated += null, default(int));
|
||||||
|
|
||||||
|
Assert.AreEqual(2, eventCount);
|
||||||
|
Assert.AreEqual(0, sut.GetWindows().Count());
|
||||||
|
}
|
||||||
|
|
||||||
|
[TestMethod]
|
||||||
|
public void Terminate_MustStopAllInstancesCorrectly()
|
||||||
|
{
|
||||||
|
var process1 = new Mock<IProcess>();
|
||||||
|
var process2 = new Mock<IProcess>();
|
||||||
|
|
||||||
|
process1.Setup(p => p.TryClose(It.IsAny<int>())).Returns(false);
|
||||||
|
process1.Setup(p => p.TryKill(It.IsAny<int>())).Returns(true);
|
||||||
|
process1.SetupGet(p => p.Id).Returns(1234);
|
||||||
|
process2.Setup(p => p.TryClose(It.IsAny<int>())).Returns(true);
|
||||||
|
process2.SetupGet(p => p.Id).Returns(5678);
|
||||||
|
processFactory.Setup(f => f.StartNew(It.IsAny<string>(), It.IsAny<string[]>())).Returns(process1.Object);
|
||||||
|
|
||||||
|
sut.Initialize();
|
||||||
|
sut.Start();
|
||||||
|
|
||||||
|
applicationMonitor.Raise(m => m.InstanceStarted += null, sut.Id, process2.Object);
|
||||||
|
sut.Terminate();
|
||||||
|
|
||||||
|
process1.Verify(p => p.TryClose(It.IsAny<int>()), Times.AtLeastOnce);
|
||||||
|
process1.Verify(p => p.TryKill(It.IsAny<int>()), Times.Once);
|
||||||
|
process2.Verify(p => p.TryClose(It.IsAny<int>()), Times.Once);
|
||||||
|
process2.Verify(p => p.TryKill(It.IsAny<int>()), Times.Never);
|
||||||
|
}
|
||||||
|
|
||||||
|
[TestMethod]
|
||||||
|
public void Terminate_MustHandleFailureGracefully()
|
||||||
|
{
|
||||||
|
var process = new Mock<IProcess>();
|
||||||
|
|
||||||
|
process.Setup(p => p.TryClose(It.IsAny<int>())).Throws<Exception>();
|
||||||
|
processFactory.Setup(f => f.StartNew(It.IsAny<string>(), It.IsAny<string[]>())).Returns(process.Object);
|
||||||
|
|
||||||
|
sut.Initialize();
|
||||||
|
sut.Start();
|
||||||
|
sut.Terminate();
|
||||||
|
|
||||||
|
process.Verify(p => p.TryClose(It.IsAny<int>()), Times.AtLeastOnce);
|
||||||
|
process.Verify(p => p.TryKill(It.IsAny<int>()), Times.Never);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,64 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
|
*
|
||||||
|
* 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 Microsoft.VisualStudio.TestTools.UnitTesting;
|
||||||
|
using Moq;
|
||||||
|
using SafeExamBrowser.Core.Contracts.Resources.Icons;
|
||||||
|
using SafeExamBrowser.WindowsApi.Contracts;
|
||||||
|
|
||||||
|
namespace SafeExamBrowser.Applications.UnitTests
|
||||||
|
{
|
||||||
|
[TestClass]
|
||||||
|
public class ExternalApplicationWindowTests
|
||||||
|
{
|
||||||
|
private IntPtr handle;
|
||||||
|
private NativeIconResource icon;
|
||||||
|
private Mock<INativeMethods> nativeMethods;
|
||||||
|
|
||||||
|
private ExternalApplicationWindow sut;
|
||||||
|
|
||||||
|
[TestInitialize]
|
||||||
|
public void Initialize()
|
||||||
|
{
|
||||||
|
handle = new IntPtr(123);
|
||||||
|
icon = new NativeIconResource();
|
||||||
|
nativeMethods = new Mock<INativeMethods>();
|
||||||
|
|
||||||
|
sut = new ExternalApplicationWindow(icon, nativeMethods.Object, handle);
|
||||||
|
}
|
||||||
|
|
||||||
|
[TestMethod]
|
||||||
|
public void Activate_MustCorrectlyActivateWindow()
|
||||||
|
{
|
||||||
|
sut.Activate();
|
||||||
|
nativeMethods.Verify(n => n.ActivateWindow(It.Is<IntPtr>(h => h == handle)));
|
||||||
|
}
|
||||||
|
|
||||||
|
[TestMethod]
|
||||||
|
public void Update_MustCorrectlyUpdateWindow()
|
||||||
|
{
|
||||||
|
var iconChanged = false;
|
||||||
|
var titleChanged = false;
|
||||||
|
|
||||||
|
nativeMethods.Setup(m => m.GetWindowIcon(It.IsAny<IntPtr>())).Returns(new IntPtr(456));
|
||||||
|
nativeMethods.Setup(m => m.GetWindowTitle((It.IsAny<IntPtr>()))).Returns("Some New Window Title");
|
||||||
|
|
||||||
|
sut.IconChanged += (_) => iconChanged = true;
|
||||||
|
sut.TitleChanged += (_) => titleChanged = true;
|
||||||
|
|
||||||
|
sut.Update();
|
||||||
|
|
||||||
|
nativeMethods.Verify(m => m.GetWindowIcon(handle), Times.Once);
|
||||||
|
nativeMethods.Verify(m => m.GetWindowTitle(handle), Times.Once);
|
||||||
|
|
||||||
|
Assert.IsTrue(iconChanged);
|
||||||
|
Assert.IsTrue(titleChanged);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,16 @@
|
||||||
|
using System.Reflection;
|
||||||
|
using System.Runtime.InteropServices;
|
||||||
|
|
||||||
|
[assembly: AssemblyTitle("SafeExamBrowser.Applications.UnitTests")]
|
||||||
|
[assembly: AssemblyDescription("Safe Exam Browser")]
|
||||||
|
[assembly: AssemblyCompany("ETH Zürich")]
|
||||||
|
[assembly: AssemblyProduct("SafeExamBrowser.Applications.UnitTests")]
|
||||||
|
[assembly: AssemblyCopyright("Copyright © 2024 ETH Zürich, IT Services")]
|
||||||
|
|
||||||
|
[assembly: ComVisible(false)]
|
||||||
|
|
||||||
|
[assembly: Guid("fc6d80ec-8611-4287-87e2-17c028a10858")]
|
||||||
|
|
||||||
|
[assembly: AssemblyVersion("1.0.0.0")]
|
||||||
|
[assembly: AssemblyFileVersion("1.0.0.0")]
|
||||||
|
[assembly: AssemblyInformationalVersion("1.0.0.0")]
|
|
@ -0,0 +1,199 @@
|
||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<Project ToolsVersion="15.0" DefaultTargets="Build" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
|
||||||
|
<Import Project="..\packages\MSTest.TestAdapter.3.2.2\build\net462\MSTest.TestAdapter.props" Condition="Exists('..\packages\MSTest.TestAdapter.3.2.2\build\net462\MSTest.TestAdapter.props')" />
|
||||||
|
<Import Project="..\packages\Microsoft.Testing.Extensions.Telemetry.1.0.2\build\netstandard2.0\Microsoft.Testing.Extensions.Telemetry.props" Condition="Exists('..\packages\Microsoft.Testing.Extensions.Telemetry.1.0.2\build\netstandard2.0\Microsoft.Testing.Extensions.Telemetry.props')" />
|
||||||
|
<Import Project="..\packages\Microsoft.Testing.Platform.MSBuild.1.0.2\build\netstandard2.0\Microsoft.Testing.Platform.MSBuild.props" Condition="Exists('..\packages\Microsoft.Testing.Platform.MSBuild.1.0.2\build\netstandard2.0\Microsoft.Testing.Platform.MSBuild.props')" />
|
||||||
|
<Import Project="$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props" Condition="Exists('$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props')" />
|
||||||
|
<PropertyGroup>
|
||||||
|
<Configuration Condition=" '$(Configuration)' == '' ">Debug</Configuration>
|
||||||
|
<Platform Condition=" '$(Platform)' == '' ">AnyCPU</Platform>
|
||||||
|
<ProjectGuid>{FC6D80EC-8611-4287-87E2-17C028A10858}</ProjectGuid>
|
||||||
|
<OutputType>Library</OutputType>
|
||||||
|
<AppDesignerFolder>Properties</AppDesignerFolder>
|
||||||
|
<RootNamespace>SafeExamBrowser.Applications.UnitTests</RootNamespace>
|
||||||
|
<AssemblyName>SafeExamBrowser.Applications.UnitTests</AssemblyName>
|
||||||
|
<TargetFrameworkVersion>v4.8</TargetFrameworkVersion>
|
||||||
|
<FileAlignment>512</FileAlignment>
|
||||||
|
<ProjectTypeGuids>{3AC096D0-A1C2-E12C-1390-A8335801FDAB};{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}</ProjectTypeGuids>
|
||||||
|
<VisualStudioVersion Condition="'$(VisualStudioVersion)' == ''">15.0</VisualStudioVersion>
|
||||||
|
<VSToolsPath Condition="'$(VSToolsPath)' == ''">$(MSBuildExtensionsPath32)\Microsoft\VisualStudio\v$(VisualStudioVersion)</VSToolsPath>
|
||||||
|
<ReferencePath>$(ProgramFiles)\Common Files\microsoft shared\VSTT\$(VisualStudioVersion)\UITestExtensionPackages</ReferencePath>
|
||||||
|
<IsCodedUITest>False</IsCodedUITest>
|
||||||
|
<TestProjectType>UnitTest</TestProjectType>
|
||||||
|
<NuGetPackageImportStamp>
|
||||||
|
</NuGetPackageImportStamp>
|
||||||
|
<TargetFrameworkProfile />
|
||||||
|
</PropertyGroup>
|
||||||
|
<PropertyGroup Condition="'$(Configuration)|$(Platform)' == 'Debug|x64'">
|
||||||
|
<DebugSymbols>true</DebugSymbols>
|
||||||
|
<OutputPath>bin\x64\Debug\</OutputPath>
|
||||||
|
<DefineConstants>DEBUG;TRACE</DefineConstants>
|
||||||
|
<DebugType>full</DebugType>
|
||||||
|
<PlatformTarget>x64</PlatformTarget>
|
||||||
|
<LangVersion>7.3</LangVersion>
|
||||||
|
<ErrorReport>prompt</ErrorReport>
|
||||||
|
</PropertyGroup>
|
||||||
|
<PropertyGroup Condition="'$(Configuration)|$(Platform)' == 'Release|x64'">
|
||||||
|
<OutputPath>bin\x64\Release\</OutputPath>
|
||||||
|
<DefineConstants>TRACE</DefineConstants>
|
||||||
|
<Optimize>true</Optimize>
|
||||||
|
<DebugType>pdbonly</DebugType>
|
||||||
|
<PlatformTarget>x64</PlatformTarget>
|
||||||
|
<LangVersion>7.3</LangVersion>
|
||||||
|
<ErrorReport>prompt</ErrorReport>
|
||||||
|
</PropertyGroup>
|
||||||
|
<PropertyGroup Condition="'$(Configuration)|$(Platform)' == 'Debug|x86'">
|
||||||
|
<DebugSymbols>true</DebugSymbols>
|
||||||
|
<OutputPath>bin\x86\Debug\</OutputPath>
|
||||||
|
<DefineConstants>DEBUG;TRACE</DefineConstants>
|
||||||
|
<DebugType>full</DebugType>
|
||||||
|
<PlatformTarget>x86</PlatformTarget>
|
||||||
|
<LangVersion>7.3</LangVersion>
|
||||||
|
<ErrorReport>prompt</ErrorReport>
|
||||||
|
</PropertyGroup>
|
||||||
|
<PropertyGroup Condition="'$(Configuration)|$(Platform)' == 'Release|x86'">
|
||||||
|
<OutputPath>bin\x86\Release\</OutputPath>
|
||||||
|
<DefineConstants>TRACE</DefineConstants>
|
||||||
|
<Optimize>true</Optimize>
|
||||||
|
<DebugType>pdbonly</DebugType>
|
||||||
|
<PlatformTarget>x86</PlatformTarget>
|
||||||
|
<LangVersion>7.3</LangVersion>
|
||||||
|
<ErrorReport>prompt</ErrorReport>
|
||||||
|
</PropertyGroup>
|
||||||
|
<ItemGroup>
|
||||||
|
<Reference Include="Castle.Core, Version=5.0.0.0, Culture=neutral, PublicKeyToken=407dd0808d44fbdc, processorArchitecture=MSIL">
|
||||||
|
<HintPath>..\packages\Castle.Core.5.1.1\lib\net462\Castle.Core.dll</HintPath>
|
||||||
|
</Reference>
|
||||||
|
<Reference Include="Microsoft.ApplicationInsights, Version=2.22.0.997, Culture=neutral, PublicKeyToken=31bf3856ad364e35, processorArchitecture=MSIL">
|
||||||
|
<HintPath>..\packages\Microsoft.ApplicationInsights.2.22.0\lib\net46\Microsoft.ApplicationInsights.dll</HintPath>
|
||||||
|
</Reference>
|
||||||
|
<Reference Include="Microsoft.CSharp" />
|
||||||
|
<Reference Include="Microsoft.Testing.Extensions.Telemetry, Version=1.0.2.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL">
|
||||||
|
<HintPath>..\packages\Microsoft.Testing.Extensions.Telemetry.1.0.2\lib\netstandard2.0\Microsoft.Testing.Extensions.Telemetry.dll</HintPath>
|
||||||
|
</Reference>
|
||||||
|
<Reference Include="Microsoft.Testing.Extensions.TrxReport.Abstractions, Version=1.0.2.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL">
|
||||||
|
<HintPath>..\packages\Microsoft.Testing.Extensions.TrxReport.Abstractions.1.0.2\lib\netstandard2.0\Microsoft.Testing.Extensions.TrxReport.Abstractions.dll</HintPath>
|
||||||
|
</Reference>
|
||||||
|
<Reference Include="Microsoft.Testing.Extensions.VSTestBridge, Version=1.0.2.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL">
|
||||||
|
<HintPath>..\packages\Microsoft.Testing.Extensions.VSTestBridge.1.0.2\lib\netstandard2.0\Microsoft.Testing.Extensions.VSTestBridge.dll</HintPath>
|
||||||
|
</Reference>
|
||||||
|
<Reference Include="Microsoft.Testing.Platform, Version=1.0.2.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL">
|
||||||
|
<HintPath>..\packages\Microsoft.Testing.Platform.MSBuild.1.0.2\lib\netstandard2.0\Microsoft.Testing.Platform.dll</HintPath>
|
||||||
|
</Reference>
|
||||||
|
<Reference Include="Microsoft.Testing.Platform.MSBuild, Version=1.0.2.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL">
|
||||||
|
<HintPath>..\packages\Microsoft.Testing.Platform.MSBuild.1.0.2\lib\netstandard2.0\Microsoft.Testing.Platform.MSBuild.dll</HintPath>
|
||||||
|
</Reference>
|
||||||
|
<Reference Include="Microsoft.TestPlatform.CoreUtilities, Version=15.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL">
|
||||||
|
<HintPath>..\packages\Microsoft.TestPlatform.ObjectModel.17.9.0\lib\net462\Microsoft.TestPlatform.CoreUtilities.dll</HintPath>
|
||||||
|
</Reference>
|
||||||
|
<Reference Include="Microsoft.TestPlatform.PlatformAbstractions, Version=15.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL">
|
||||||
|
<HintPath>..\packages\Microsoft.TestPlatform.ObjectModel.17.9.0\lib\net462\Microsoft.TestPlatform.PlatformAbstractions.dll</HintPath>
|
||||||
|
</Reference>
|
||||||
|
<Reference Include="Microsoft.VisualStudio.TestPlatform.ObjectModel, Version=15.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL">
|
||||||
|
<HintPath>..\packages\Microsoft.TestPlatform.ObjectModel.17.9.0\lib\net462\Microsoft.VisualStudio.TestPlatform.ObjectModel.dll</HintPath>
|
||||||
|
</Reference>
|
||||||
|
<Reference Include="Microsoft.VisualStudio.TestPlatform.TestFramework, Version=14.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL">
|
||||||
|
<HintPath>..\packages\MSTest.TestFramework.3.2.2\lib\net462\Microsoft.VisualStudio.TestPlatform.TestFramework.dll</HintPath>
|
||||||
|
</Reference>
|
||||||
|
<Reference Include="Microsoft.VisualStudio.TestPlatform.TestFramework.Extensions, Version=14.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL">
|
||||||
|
<HintPath>..\packages\MSTest.TestFramework.3.2.2\lib\net462\Microsoft.VisualStudio.TestPlatform.TestFramework.Extensions.dll</HintPath>
|
||||||
|
</Reference>
|
||||||
|
<Reference Include="Moq, Version=4.20.70.0, Culture=neutral, PublicKeyToken=69f491c39445e920, processorArchitecture=MSIL">
|
||||||
|
<HintPath>..\packages\Moq.4.20.70\lib\net462\Moq.dll</HintPath>
|
||||||
|
</Reference>
|
||||||
|
<Reference Include="NuGet.Frameworks, Version=6.9.1.3, Culture=neutral, PublicKeyToken=31bf3856ad364e35, processorArchitecture=MSIL">
|
||||||
|
<HintPath>..\packages\NuGet.Frameworks.6.9.1\lib\net472\NuGet.Frameworks.dll</HintPath>
|
||||||
|
</Reference>
|
||||||
|
<Reference Include="System" />
|
||||||
|
<Reference Include="System.Buffers, Version=4.0.3.0, Culture=neutral, PublicKeyToken=cc7b13ffcd2ddd51, processorArchitecture=MSIL">
|
||||||
|
<HintPath>..\packages\System.Buffers.4.5.1\lib\net461\System.Buffers.dll</HintPath>
|
||||||
|
</Reference>
|
||||||
|
<Reference Include="System.Collections.Immutable, Version=8.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL">
|
||||||
|
<HintPath>..\packages\System.Collections.Immutable.8.0.0\lib\net462\System.Collections.Immutable.dll</HintPath>
|
||||||
|
</Reference>
|
||||||
|
<Reference Include="System.Configuration" />
|
||||||
|
<Reference Include="System.Core" />
|
||||||
|
<Reference Include="System.Diagnostics.DiagnosticSource, Version=8.0.0.0, Culture=neutral, PublicKeyToken=cc7b13ffcd2ddd51, processorArchitecture=MSIL">
|
||||||
|
<HintPath>..\packages\System.Diagnostics.DiagnosticSource.8.0.0\lib\net462\System.Diagnostics.DiagnosticSource.dll</HintPath>
|
||||||
|
</Reference>
|
||||||
|
<Reference Include="System.Memory, Version=4.0.1.2, Culture=neutral, PublicKeyToken=cc7b13ffcd2ddd51, processorArchitecture=MSIL">
|
||||||
|
<HintPath>..\packages\System.Memory.4.5.5\lib\net461\System.Memory.dll</HintPath>
|
||||||
|
</Reference>
|
||||||
|
<Reference Include="System.Net.Http" />
|
||||||
|
<Reference Include="System.Numerics" />
|
||||||
|
<Reference Include="System.Numerics.Vectors, Version=4.1.4.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL">
|
||||||
|
<HintPath>..\packages\System.Numerics.Vectors.4.5.0\lib\net46\System.Numerics.Vectors.dll</HintPath>
|
||||||
|
</Reference>
|
||||||
|
<Reference Include="System.Reflection.Metadata, Version=8.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL">
|
||||||
|
<HintPath>..\packages\System.Reflection.Metadata.8.0.0\lib\net462\System.Reflection.Metadata.dll</HintPath>
|
||||||
|
</Reference>
|
||||||
|
<Reference Include="System.Runtime" />
|
||||||
|
<Reference Include="System.Runtime.CompilerServices.Unsafe, Version=6.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL">
|
||||||
|
<HintPath>..\packages\System.Runtime.CompilerServices.Unsafe.6.0.0\lib\net461\System.Runtime.CompilerServices.Unsafe.dll</HintPath>
|
||||||
|
</Reference>
|
||||||
|
<Reference Include="System.Runtime.Serialization" />
|
||||||
|
<Reference Include="System.Threading.Tasks.Extensions, Version=4.2.0.1, Culture=neutral, PublicKeyToken=cc7b13ffcd2ddd51, processorArchitecture=MSIL">
|
||||||
|
<HintPath>..\packages\System.Threading.Tasks.Extensions.4.5.4\lib\net461\System.Threading.Tasks.Extensions.dll</HintPath>
|
||||||
|
</Reference>
|
||||||
|
<Reference Include="System.Xml" />
|
||||||
|
</ItemGroup>
|
||||||
|
<ItemGroup>
|
||||||
|
<Compile Include="ApplicationFactoryTests.cs" />
|
||||||
|
<Compile Include="Properties\AssemblyInfo.cs" />
|
||||||
|
<Compile Include="ExternalApplicationTests.cs" />
|
||||||
|
<Compile Include="ExternalApplicationWindowTests.cs" />
|
||||||
|
<Compile Include="ExternalApplicationInstanceTests.cs" />
|
||||||
|
</ItemGroup>
|
||||||
|
<ItemGroup>
|
||||||
|
<None Include="app.config" />
|
||||||
|
<None Include="packages.config" />
|
||||||
|
</ItemGroup>
|
||||||
|
<ItemGroup>
|
||||||
|
<ProjectReference Include="..\SafeExamBrowser.Applications.Contracts\SafeExamBrowser.Applications.Contracts.csproj">
|
||||||
|
<Project>{ac77745d-3b41-43e2-8e84-d40e5a4ee77f}</Project>
|
||||||
|
<Name>SafeExamBrowser.Applications.Contracts</Name>
|
||||||
|
</ProjectReference>
|
||||||
|
<ProjectReference Include="..\SafeExamBrowser.Applications\SafeExamBrowser.Applications.csproj">
|
||||||
|
<Project>{a113e68f-1209-4689-981a-15c554b2df4e}</Project>
|
||||||
|
<Name>SafeExamBrowser.Applications</Name>
|
||||||
|
</ProjectReference>
|
||||||
|
<ProjectReference Include="..\SafeExamBrowser.Core.Contracts\SafeExamBrowser.Core.Contracts.csproj">
|
||||||
|
<Project>{fe0e1224-b447-4b14-81e7-ed7d84822aa0}</Project>
|
||||||
|
<Name>SafeExamBrowser.Core.Contracts</Name>
|
||||||
|
</ProjectReference>
|
||||||
|
<ProjectReference Include="..\SafeExamBrowser.Logging.Contracts\SafeExamBrowser.Logging.Contracts.csproj">
|
||||||
|
<Project>{64ea30fb-11d4-436a-9c2b-88566285363e}</Project>
|
||||||
|
<Name>SafeExamBrowser.Logging.Contracts</Name>
|
||||||
|
</ProjectReference>
|
||||||
|
<ProjectReference Include="..\SafeExamBrowser.Monitoring.Contracts\SafeExamBrowser.Monitoring.Contracts.csproj">
|
||||||
|
<Project>{6d563a30-366d-4c35-815b-2c9e6872278b}</Project>
|
||||||
|
<Name>SafeExamBrowser.Monitoring.Contracts</Name>
|
||||||
|
</ProjectReference>
|
||||||
|
<ProjectReference Include="..\SafeExamBrowser.Settings\SafeExamBrowser.Settings.csproj">
|
||||||
|
<Project>{30b2d907-5861-4f39-abad-c4abf1b3470e}</Project>
|
||||||
|
<Name>SafeExamBrowser.Settings</Name>
|
||||||
|
</ProjectReference>
|
||||||
|
<ProjectReference Include="..\SafeExamBrowser.SystemComponents.Contracts\SafeExamBrowser.SystemComponents.Contracts.csproj">
|
||||||
|
<Project>{903129c6-e236-493b-9ad6-c6a57f647a3a}</Project>
|
||||||
|
<Name>SafeExamBrowser.SystemComponents.Contracts</Name>
|
||||||
|
</ProjectReference>
|
||||||
|
<ProjectReference Include="..\SafeExamBrowser.WindowsApi.Contracts\SafeExamBrowser.WindowsApi.Contracts.csproj">
|
||||||
|
<Project>{7016f080-9aa5-41b2-a225-385ad877c171}</Project>
|
||||||
|
<Name>SafeExamBrowser.WindowsApi.Contracts</Name>
|
||||||
|
</ProjectReference>
|
||||||
|
</ItemGroup>
|
||||||
|
<Import Project="$(VSToolsPath)\TeamTest\Microsoft.TestTools.targets" Condition="Exists('$(VSToolsPath)\TeamTest\Microsoft.TestTools.targets')" />
|
||||||
|
<Import Project="$(MSBuildToolsPath)\Microsoft.CSharp.targets" />
|
||||||
|
<Target Name="EnsureNuGetPackageBuildImports" BeforeTargets="PrepareForBuild">
|
||||||
|
<PropertyGroup>
|
||||||
|
<ErrorText>This project references NuGet package(s) that are missing on this computer. Use NuGet Package Restore to download them. For more information, see http://go.microsoft.com/fwlink/?LinkID=322105. The missing file is {0}.</ErrorText>
|
||||||
|
</PropertyGroup>
|
||||||
|
<Error Condition="!Exists('..\packages\Microsoft.Testing.Platform.MSBuild.1.0.2\build\netstandard2.0\Microsoft.Testing.Platform.MSBuild.props')" Text="$([System.String]::Format('$(ErrorText)', '..\packages\Microsoft.Testing.Platform.MSBuild.1.0.2\build\netstandard2.0\Microsoft.Testing.Platform.MSBuild.props'))" />
|
||||||
|
<Error Condition="!Exists('..\packages\Microsoft.Testing.Platform.MSBuild.1.0.2\build\netstandard2.0\Microsoft.Testing.Platform.MSBuild.targets')" Text="$([System.String]::Format('$(ErrorText)', '..\packages\Microsoft.Testing.Platform.MSBuild.1.0.2\build\netstandard2.0\Microsoft.Testing.Platform.MSBuild.targets'))" />
|
||||||
|
<Error Condition="!Exists('..\packages\Microsoft.Testing.Extensions.Telemetry.1.0.2\build\netstandard2.0\Microsoft.Testing.Extensions.Telemetry.props')" Text="$([System.String]::Format('$(ErrorText)', '..\packages\Microsoft.Testing.Extensions.Telemetry.1.0.2\build\netstandard2.0\Microsoft.Testing.Extensions.Telemetry.props'))" />
|
||||||
|
<Error Condition="!Exists('..\packages\MSTest.TestAdapter.3.2.2\build\net462\MSTest.TestAdapter.props')" Text="$([System.String]::Format('$(ErrorText)', '..\packages\MSTest.TestAdapter.3.2.2\build\net462\MSTest.TestAdapter.props'))" />
|
||||||
|
<Error Condition="!Exists('..\packages\MSTest.TestAdapter.3.2.2\build\net462\MSTest.TestAdapter.targets')" Text="$([System.String]::Format('$(ErrorText)', '..\packages\MSTest.TestAdapter.3.2.2\build\net462\MSTest.TestAdapter.targets'))" />
|
||||||
|
</Target>
|
||||||
|
<Import Project="..\packages\Microsoft.Testing.Platform.MSBuild.1.0.2\build\netstandard2.0\Microsoft.Testing.Platform.MSBuild.targets" Condition="Exists('..\packages\Microsoft.Testing.Platform.MSBuild.1.0.2\build\netstandard2.0\Microsoft.Testing.Platform.MSBuild.targets')" />
|
||||||
|
<Import Project="..\packages\MSTest.TestAdapter.3.2.2\build\net462\MSTest.TestAdapter.targets" Condition="Exists('..\packages\MSTest.TestAdapter.3.2.2\build\net462\MSTest.TestAdapter.targets')" />
|
||||||
|
</Project>
|
35
SafeExamBrowser.Applications.UnitTests/app.config
Normal file
35
SafeExamBrowser.Applications.UnitTests/app.config
Normal file
|
@ -0,0 +1,35 @@
|
||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<configuration>
|
||||||
|
<runtime>
|
||||||
|
<assemblyBinding xmlns="urn:schemas-microsoft-com:asm.v1">
|
||||||
|
<dependentAssembly>
|
||||||
|
<assemblyIdentity name="System.Runtime.CompilerServices.Unsafe" publicKeyToken="b03f5f7f11d50a3a" culture="neutral" />
|
||||||
|
<bindingRedirect oldVersion="0.0.0.0-6.0.0.0" newVersion="6.0.0.0" />
|
||||||
|
</dependentAssembly>
|
||||||
|
<dependentAssembly>
|
||||||
|
<assemblyIdentity name="NuGet.Frameworks" publicKeyToken="31bf3856ad364e35" culture="neutral" />
|
||||||
|
<bindingRedirect oldVersion="0.0.0.0-5.11.3.1" newVersion="5.11.3.1" />
|
||||||
|
</dependentAssembly>
|
||||||
|
<dependentAssembly>
|
||||||
|
<assemblyIdentity name="System.Collections.Immutable" publicKeyToken="b03f5f7f11d50a3a" culture="neutral" />
|
||||||
|
<bindingRedirect oldVersion="0.0.0.0-8.0.0.0" newVersion="8.0.0.0" />
|
||||||
|
</dependentAssembly>
|
||||||
|
<dependentAssembly>
|
||||||
|
<assemblyIdentity name="Microsoft.ApplicationInsights" publicKeyToken="31bf3856ad364e35" culture="neutral" />
|
||||||
|
<bindingRedirect oldVersion="0.0.0.0-2.22.0.997" newVersion="2.22.0.997" />
|
||||||
|
</dependentAssembly>
|
||||||
|
<dependentAssembly>
|
||||||
|
<assemblyIdentity name="System.Diagnostics.DiagnosticSource" publicKeyToken="cc7b13ffcd2ddd51" culture="neutral" />
|
||||||
|
<bindingRedirect oldVersion="0.0.0.0-8.0.0.0" newVersion="8.0.0.0" />
|
||||||
|
</dependentAssembly>
|
||||||
|
<dependentAssembly>
|
||||||
|
<assemblyIdentity name="System.Memory" publicKeyToken="cc7b13ffcd2ddd51" culture="neutral" />
|
||||||
|
<bindingRedirect oldVersion="0.0.0.0-4.0.1.2" newVersion="4.0.1.2" />
|
||||||
|
</dependentAssembly>
|
||||||
|
<dependentAssembly>
|
||||||
|
<assemblyIdentity name="System.Reflection.Metadata" publicKeyToken="b03f5f7f11d50a3a" culture="neutral" />
|
||||||
|
<bindingRedirect oldVersion="0.0.0.0-8.0.0.0" newVersion="8.0.0.0" />
|
||||||
|
</dependentAssembly>
|
||||||
|
</assemblyBinding>
|
||||||
|
</runtime>
|
||||||
|
<startup><supportedRuntime version="v4.0" sku=".NETFramework,Version=v4.8" /></startup></configuration>
|
23
SafeExamBrowser.Applications.UnitTests/packages.config
Normal file
23
SafeExamBrowser.Applications.UnitTests/packages.config
Normal file
|
@ -0,0 +1,23 @@
|
||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<packages>
|
||||||
|
<package id="Castle.Core" version="5.1.1" targetFramework="net48" />
|
||||||
|
<package id="Microsoft.ApplicationInsights" version="2.22.0" targetFramework="net48" />
|
||||||
|
<package id="Microsoft.Testing.Extensions.Telemetry" version="1.0.2" targetFramework="net48" />
|
||||||
|
<package id="Microsoft.Testing.Extensions.TrxReport.Abstractions" version="1.0.2" targetFramework="net48" />
|
||||||
|
<package id="Microsoft.Testing.Extensions.VSTestBridge" version="1.0.2" targetFramework="net48" />
|
||||||
|
<package id="Microsoft.Testing.Platform" version="1.0.2" targetFramework="net48" />
|
||||||
|
<package id="Microsoft.Testing.Platform.MSBuild" version="1.0.2" targetFramework="net48" />
|
||||||
|
<package id="Microsoft.TestPlatform.ObjectModel" version="17.9.0" targetFramework="net48" />
|
||||||
|
<package id="Moq" version="4.20.70" targetFramework="net48" />
|
||||||
|
<package id="MSTest.TestAdapter" version="3.2.2" targetFramework="net48" />
|
||||||
|
<package id="MSTest.TestFramework" version="3.2.2" targetFramework="net48" />
|
||||||
|
<package id="NuGet.Frameworks" version="6.9.1" targetFramework="net48" />
|
||||||
|
<package id="System.Buffers" version="4.5.1" targetFramework="net48" />
|
||||||
|
<package id="System.Collections.Immutable" version="8.0.0" targetFramework="net48" />
|
||||||
|
<package id="System.Diagnostics.DiagnosticSource" version="8.0.0" targetFramework="net48" />
|
||||||
|
<package id="System.Memory" version="4.5.5" targetFramework="net48" />
|
||||||
|
<package id="System.Numerics.Vectors" version="4.5.0" targetFramework="net48" />
|
||||||
|
<package id="System.Reflection.Metadata" version="8.0.0" targetFramework="net48" />
|
||||||
|
<package id="System.Runtime.CompilerServices.Unsafe" version="6.0.0" targetFramework="net48" />
|
||||||
|
<package id="System.Threading.Tasks.Extensions" version="4.5.4" targetFramework="net48" />
|
||||||
|
</packages>
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
@ -9,39 +9,42 @@
|
||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.IO;
|
using System.IO;
|
||||||
using Microsoft.Win32;
|
|
||||||
using SafeExamBrowser.Applications.Contracts;
|
using SafeExamBrowser.Applications.Contracts;
|
||||||
using SafeExamBrowser.Logging.Contracts;
|
using SafeExamBrowser.Logging.Contracts;
|
||||||
using SafeExamBrowser.Monitoring.Contracts.Applications;
|
using SafeExamBrowser.Monitoring.Contracts.Applications;
|
||||||
using SafeExamBrowser.Settings.Applications;
|
using SafeExamBrowser.Settings.Applications;
|
||||||
|
using SafeExamBrowser.SystemComponents.Contracts.Registry;
|
||||||
using SafeExamBrowser.WindowsApi.Contracts;
|
using SafeExamBrowser.WindowsApi.Contracts;
|
||||||
|
|
||||||
namespace SafeExamBrowser.Applications
|
namespace SafeExamBrowser.Applications
|
||||||
{
|
{
|
||||||
public class ApplicationFactory : IApplicationFactory
|
public class ApplicationFactory : IApplicationFactory
|
||||||
{
|
{
|
||||||
private IApplicationMonitor applicationMonitor;
|
private readonly IApplicationMonitor applicationMonitor;
|
||||||
private IModuleLogger logger;
|
private readonly IModuleLogger logger;
|
||||||
private INativeMethods nativeMethods;
|
private readonly INativeMethods nativeMethods;
|
||||||
private IProcessFactory processFactory;
|
private readonly IProcessFactory processFactory;
|
||||||
|
private readonly IRegistry registry;
|
||||||
|
|
||||||
public ApplicationFactory(
|
public ApplicationFactory(
|
||||||
IApplicationMonitor applicationMonitor,
|
IApplicationMonitor applicationMonitor,
|
||||||
IModuleLogger logger,
|
IModuleLogger logger,
|
||||||
INativeMethods nativeMethods,
|
INativeMethods nativeMethods,
|
||||||
IProcessFactory processFactory)
|
IProcessFactory processFactory,
|
||||||
|
IRegistry registry)
|
||||||
{
|
{
|
||||||
this.applicationMonitor = applicationMonitor;
|
this.applicationMonitor = applicationMonitor;
|
||||||
this.logger = logger;
|
this.logger = logger;
|
||||||
this.nativeMethods = nativeMethods;
|
this.nativeMethods = nativeMethods;
|
||||||
this.processFactory = processFactory;
|
this.processFactory = processFactory;
|
||||||
|
this.registry = registry;
|
||||||
}
|
}
|
||||||
|
|
||||||
public FactoryResult TryCreate(WhitelistApplication settings, out IApplication application)
|
public FactoryResult TryCreate(WhitelistApplication settings, out IApplication<IApplicationWindow> application)
|
||||||
{
|
{
|
||||||
var name = $"'{settings.DisplayName}' ({ settings.ExecutableName})";
|
var name = $"'{settings.DisplayName}' ({settings.ExecutableName})";
|
||||||
|
|
||||||
application = default(IApplication);
|
application = default;
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
|
@ -69,10 +72,12 @@ namespace SafeExamBrowser.Applications
|
||||||
return FactoryResult.Error;
|
return FactoryResult.Error;
|
||||||
}
|
}
|
||||||
|
|
||||||
private IApplication BuildApplication(string executablePath, WhitelistApplication settings)
|
private IApplication<IApplicationWindow> BuildApplication(string executablePath, WhitelistApplication settings)
|
||||||
{
|
{
|
||||||
|
const int ONE_SECOND = 1000;
|
||||||
|
|
||||||
var applicationLogger = logger.CloneFor(settings.DisplayName);
|
var applicationLogger = logger.CloneFor(settings.DisplayName);
|
||||||
var application = new ExternalApplication(applicationMonitor, executablePath, applicationLogger, nativeMethods, processFactory, settings);
|
var application = new ExternalApplication(applicationMonitor, executablePath, applicationLogger, nativeMethods, processFactory, settings, ONE_SECOND);
|
||||||
|
|
||||||
return application;
|
return application;
|
||||||
}
|
}
|
||||||
|
@ -82,14 +87,14 @@ namespace SafeExamBrowser.Applications
|
||||||
var paths = new List<string[]>();
|
var paths = new List<string[]>();
|
||||||
var registryPath = QueryPathFromRegistry(settings);
|
var registryPath = QueryPathFromRegistry(settings);
|
||||||
|
|
||||||
mainExecutable = default(string);
|
mainExecutable = default;
|
||||||
|
|
||||||
paths.Add(new[] { Environment.GetFolderPath(Environment.SpecialFolder.ProgramFiles), settings.ExecutableName });
|
paths.Add(new[] { Environment.GetFolderPath(Environment.SpecialFolder.ProgramFiles), settings.ExecutableName });
|
||||||
paths.Add(new[] { Environment.GetFolderPath(Environment.SpecialFolder.ProgramFilesX86), settings.ExecutableName });
|
paths.Add(new[] { Environment.GetFolderPath(Environment.SpecialFolder.ProgramFilesX86), settings.ExecutableName });
|
||||||
paths.Add(new[] { Environment.GetFolderPath(Environment.SpecialFolder.System), settings.ExecutableName });
|
paths.Add(new[] { Environment.GetFolderPath(Environment.SpecialFolder.System), settings.ExecutableName });
|
||||||
paths.Add(new[] { Environment.GetFolderPath(Environment.SpecialFolder.SystemX86), settings.ExecutableName });
|
paths.Add(new[] { Environment.GetFolderPath(Environment.SpecialFolder.SystemX86), settings.ExecutableName });
|
||||||
|
|
||||||
if (settings.ExecutablePath != default(string))
|
if (settings.ExecutablePath != default)
|
||||||
{
|
{
|
||||||
paths.Add(new[] { settings.ExecutablePath, settings.ExecutableName });
|
paths.Add(new[] { settings.ExecutablePath, settings.ExecutableName });
|
||||||
paths.Add(new[] { Environment.GetFolderPath(Environment.SpecialFolder.ProgramFiles), settings.ExecutablePath, settings.ExecutableName });
|
paths.Add(new[] { Environment.GetFolderPath(Environment.SpecialFolder.ProgramFiles), settings.ExecutablePath, settings.ExecutableName });
|
||||||
|
@ -98,11 +103,11 @@ namespace SafeExamBrowser.Applications
|
||||||
paths.Add(new[] { Environment.GetFolderPath(Environment.SpecialFolder.SystemX86), settings.ExecutablePath, settings.ExecutableName });
|
paths.Add(new[] { Environment.GetFolderPath(Environment.SpecialFolder.SystemX86), settings.ExecutablePath, settings.ExecutableName });
|
||||||
}
|
}
|
||||||
|
|
||||||
if (registryPath != default(string))
|
if (registryPath != default)
|
||||||
{
|
{
|
||||||
paths.Add(new[] { registryPath, settings.ExecutableName });
|
paths.Add(new[] { registryPath, settings.ExecutableName });
|
||||||
|
|
||||||
if (settings.ExecutablePath != default(string))
|
if (settings.ExecutablePath != default)
|
||||||
{
|
{
|
||||||
paths.Add(new[] { registryPath, settings.ExecutablePath, settings.ExecutableName });
|
paths.Add(new[] { registryPath, settings.ExecutablePath, settings.ExecutableName });
|
||||||
}
|
}
|
||||||
|
@ -131,22 +136,12 @@ namespace SafeExamBrowser.Applications
|
||||||
|
|
||||||
private string QueryPathFromRegistry(WhitelistApplication settings)
|
private string QueryPathFromRegistry(WhitelistApplication settings)
|
||||||
{
|
{
|
||||||
try
|
if (registry.TryRead($@"{RegistryValue.MachineHive.AppPaths_Key}\{settings.ExecutableName}", "Path", out var value))
|
||||||
{
|
{
|
||||||
using (var key = Registry.LocalMachine.OpenSubKey($@"SOFTWARE\Microsoft\Windows\CurrentVersion\App Paths\{settings.ExecutableName}"))
|
return value as string;
|
||||||
{
|
|
||||||
if (key != null)
|
|
||||||
{
|
|
||||||
return key.GetValue("Path") as string;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
catch (Exception e)
|
|
||||||
{
|
|
||||||
logger.Error($"Failed to query path in registry for '{settings.ExecutableName}'!", e);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return default(string);
|
return default;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
@ -19,17 +19,18 @@ using SafeExamBrowser.WindowsApi.Contracts;
|
||||||
|
|
||||||
namespace SafeExamBrowser.Applications
|
namespace SafeExamBrowser.Applications
|
||||||
{
|
{
|
||||||
internal class ExternalApplication : IApplication
|
internal class ExternalApplication : IApplication<IApplicationWindow>
|
||||||
{
|
{
|
||||||
private readonly object @lock = new object();
|
private readonly object @lock = new object();
|
||||||
|
|
||||||
private IApplicationMonitor applicationMonitor;
|
private readonly IApplicationMonitor applicationMonitor;
|
||||||
private string executablePath;
|
private readonly string executablePath;
|
||||||
private IModuleLogger logger;
|
private readonly IList<ExternalApplicationInstance> instances;
|
||||||
private INativeMethods nativeMethods;
|
private readonly IModuleLogger logger;
|
||||||
private IList<ExternalApplicationInstance> instances;
|
private readonly INativeMethods nativeMethods;
|
||||||
private IProcessFactory processFactory;
|
private readonly IProcessFactory processFactory;
|
||||||
private WhitelistApplication settings;
|
private readonly WhitelistApplication settings;
|
||||||
|
private readonly int windowMonitoringInterval;
|
||||||
|
|
||||||
public bool AutoStart { get; private set; }
|
public bool AutoStart { get; private set; }
|
||||||
public IconResource Icon { get; private set; }
|
public IconResource Icon { get; private set; }
|
||||||
|
@ -45,7 +46,8 @@ namespace SafeExamBrowser.Applications
|
||||||
IModuleLogger logger,
|
IModuleLogger logger,
|
||||||
INativeMethods nativeMethods,
|
INativeMethods nativeMethods,
|
||||||
IProcessFactory processFactory,
|
IProcessFactory processFactory,
|
||||||
WhitelistApplication settings)
|
WhitelistApplication settings,
|
||||||
|
int windowMonitoringInterval_ms)
|
||||||
{
|
{
|
||||||
this.applicationMonitor = applicationMonitor;
|
this.applicationMonitor = applicationMonitor;
|
||||||
this.executablePath = executablePath;
|
this.executablePath = executablePath;
|
||||||
|
@ -54,6 +56,7 @@ namespace SafeExamBrowser.Applications
|
||||||
this.instances = new List<ExternalApplicationInstance>();
|
this.instances = new List<ExternalApplicationInstance>();
|
||||||
this.processFactory = processFactory;
|
this.processFactory = processFactory;
|
||||||
this.settings = settings;
|
this.settings = settings;
|
||||||
|
this.windowMonitoringInterval = windowMonitoringInterval_ms;
|
||||||
}
|
}
|
||||||
|
|
||||||
public IEnumerable<IApplicationWindow> GetWindows()
|
public IEnumerable<IApplicationWindow> GetWindows()
|
||||||
|
@ -89,18 +92,6 @@ namespace SafeExamBrowser.Applications
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private string[] BuildArguments()
|
|
||||||
{
|
|
||||||
var arguments = new List<string>();
|
|
||||||
|
|
||||||
foreach (var argument in settings.Arguments)
|
|
||||||
{
|
|
||||||
arguments.Add(Environment.ExpandEnvironmentVariables(argument));
|
|
||||||
}
|
|
||||||
|
|
||||||
return arguments.ToArray();
|
|
||||||
}
|
|
||||||
|
|
||||||
public void Terminate()
|
public void Terminate()
|
||||||
{
|
{
|
||||||
applicationMonitor.InstanceStarted -= ApplicationMonitor_InstanceStarted;
|
applicationMonitor.InstanceStarted -= ApplicationMonitor_InstanceStarted;
|
||||||
|
@ -153,12 +144,24 @@ namespace SafeExamBrowser.Applications
|
||||||
WindowsChanged?.Invoke();
|
WindowsChanged?.Invoke();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private string[] BuildArguments()
|
||||||
|
{
|
||||||
|
var arguments = new List<string>();
|
||||||
|
|
||||||
|
foreach (var argument in settings.Arguments)
|
||||||
|
{
|
||||||
|
arguments.Add(Environment.ExpandEnvironmentVariables(argument));
|
||||||
|
}
|
||||||
|
|
||||||
|
return arguments.ToArray();
|
||||||
|
}
|
||||||
|
|
||||||
private void InitializeInstance(IProcess process)
|
private void InitializeInstance(IProcess process)
|
||||||
{
|
{
|
||||||
lock (@lock)
|
lock (@lock)
|
||||||
{
|
{
|
||||||
var instanceLogger = logger.CloneFor($"{Name} ({process.Id})");
|
var instanceLogger = logger.CloneFor($"{Name} ({process.Id})");
|
||||||
var instance = new ExternalApplicationInstance(Icon, instanceLogger, nativeMethods, process);
|
var instance = new ExternalApplicationInstance(Icon, instanceLogger, nativeMethods, process, windowMonitoringInterval);
|
||||||
|
|
||||||
instance.Terminated += Instance_Terminated;
|
instance.Terminated += Instance_Terminated;
|
||||||
instance.WindowsChanged += () => WindowsChanged?.Invoke();
|
instance.WindowsChanged += () => WindowsChanged?.Invoke();
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
@ -12,8 +12,8 @@ using System.Linq;
|
||||||
using System.Timers;
|
using System.Timers;
|
||||||
using SafeExamBrowser.Applications.Contracts;
|
using SafeExamBrowser.Applications.Contracts;
|
||||||
using SafeExamBrowser.Applications.Contracts.Events;
|
using SafeExamBrowser.Applications.Contracts.Events;
|
||||||
using SafeExamBrowser.Core.Contracts.Resources.Icons;
|
|
||||||
using SafeExamBrowser.Applications.Events;
|
using SafeExamBrowser.Applications.Events;
|
||||||
|
using SafeExamBrowser.Core.Contracts.Resources.Icons;
|
||||||
using SafeExamBrowser.Logging.Contracts;
|
using SafeExamBrowser.Logging.Contracts;
|
||||||
using SafeExamBrowser.WindowsApi.Contracts;
|
using SafeExamBrowser.WindowsApi.Contracts;
|
||||||
|
|
||||||
|
@ -23,24 +23,32 @@ namespace SafeExamBrowser.Applications
|
||||||
{
|
{
|
||||||
private readonly object @lock = new object();
|
private readonly object @lock = new object();
|
||||||
|
|
||||||
private IconResource icon;
|
private readonly IconResource icon;
|
||||||
private ILogger logger;
|
private readonly ILogger logger;
|
||||||
private INativeMethods nativeMethods;
|
private readonly INativeMethods nativeMethods;
|
||||||
private IProcess process;
|
private readonly IProcess process;
|
||||||
|
private readonly int windowMonitoringInterval;
|
||||||
|
private readonly IList<ExternalApplicationWindow> windows;
|
||||||
|
|
||||||
private Timer timer;
|
private Timer timer;
|
||||||
private IList<ExternalApplicationWindow> windows;
|
|
||||||
|
|
||||||
internal int Id { get; private set; }
|
internal int Id { get; private set; }
|
||||||
|
|
||||||
internal event InstanceTerminatedEventHandler Terminated;
|
internal event InstanceTerminatedEventHandler Terminated;
|
||||||
internal event WindowsChangedEventHandler WindowsChanged;
|
internal event WindowsChangedEventHandler WindowsChanged;
|
||||||
|
|
||||||
internal ExternalApplicationInstance(IconResource icon, ILogger logger, INativeMethods nativeMethods, IProcess process)
|
internal ExternalApplicationInstance(
|
||||||
|
IconResource icon,
|
||||||
|
ILogger logger,
|
||||||
|
INativeMethods nativeMethods,
|
||||||
|
IProcess process,
|
||||||
|
int windowMonitoringInterval_ms)
|
||||||
{
|
{
|
||||||
this.icon = icon;
|
this.icon = icon;
|
||||||
this.logger = logger;
|
this.logger = logger;
|
||||||
this.nativeMethods = nativeMethods;
|
this.nativeMethods = nativeMethods;
|
||||||
this.process = process;
|
this.process = process;
|
||||||
|
this.windowMonitoringInterval = windowMonitoringInterval_ms;
|
||||||
this.windows = new List<ExternalApplicationWindow>();
|
this.windows = new List<ExternalApplicationWindow>();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -145,19 +153,20 @@ namespace SafeExamBrowser.Applications
|
||||||
|
|
||||||
private void InitializeEvents()
|
private void InitializeEvents()
|
||||||
{
|
{
|
||||||
const int ONE_SECOND = 1000;
|
|
||||||
|
|
||||||
process.Terminated += Process_Terminated;
|
process.Terminated += Process_Terminated;
|
||||||
|
|
||||||
timer = new Timer(ONE_SECOND);
|
timer = new Timer(windowMonitoringInterval);
|
||||||
timer.Elapsed += Timer_Elapsed;
|
timer.Elapsed += Timer_Elapsed;
|
||||||
timer.Start();
|
timer.Start();
|
||||||
}
|
}
|
||||||
|
|
||||||
private void FinalizeEvents()
|
private void FinalizeEvents()
|
||||||
|
{
|
||||||
|
if (timer != default)
|
||||||
{
|
{
|
||||||
timer.Elapsed -= Timer_Elapsed;
|
timer.Elapsed -= Timer_Elapsed;
|
||||||
timer.Stop();
|
timer.Stop();
|
||||||
|
}
|
||||||
|
|
||||||
process.Terminated -= Process_Terminated;
|
process.Terminated -= Process_Terminated;
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
@ -16,7 +16,7 @@ namespace SafeExamBrowser.Applications
|
||||||
{
|
{
|
||||||
internal class ExternalApplicationWindow : IApplicationWindow
|
internal class ExternalApplicationWindow : IApplicationWindow
|
||||||
{
|
{
|
||||||
private INativeMethods nativeMethods;
|
private readonly INativeMethods nativeMethods;
|
||||||
|
|
||||||
public IntPtr Handle { get; }
|
public IntPtr Handle { get; }
|
||||||
public IconResource Icon { get; private set; }
|
public IconResource Icon { get; private set; }
|
||||||
|
|
|
@ -1,4 +1,5 @@
|
||||||
using System.Reflection;
|
using System.Reflection;
|
||||||
|
using System.Runtime.CompilerServices;
|
||||||
using System.Runtime.InteropServices;
|
using System.Runtime.InteropServices;
|
||||||
|
|
||||||
// General Information about an assembly is controlled through the following
|
// General Information about an assembly is controlled through the following
|
||||||
|
@ -8,12 +9,13 @@ using System.Runtime.InteropServices;
|
||||||
[assembly: AssemblyDescription("Safe Exam Browser")]
|
[assembly: AssemblyDescription("Safe Exam Browser")]
|
||||||
[assembly: AssemblyCompany("ETH Zürich")]
|
[assembly: AssemblyCompany("ETH Zürich")]
|
||||||
[assembly: AssemblyProduct("SafeExamBrowser.Applications")]
|
[assembly: AssemblyProduct("SafeExamBrowser.Applications")]
|
||||||
[assembly: AssemblyCopyright("Copyright © 2023 ETH Zürich, Educational Development and Technology (LET)")]
|
[assembly: AssemblyCopyright("Copyright © 2024 ETH Zürich, IT Services")]
|
||||||
|
|
||||||
// Setting ComVisible to false makes the types in this assembly not visible
|
// Setting ComVisible to false makes the types in this assembly not visible
|
||||||
// to COM components. If you need to access a type in this assembly from
|
// to COM components. If you need to access a type in this assembly from
|
||||||
// COM, set the ComVisible attribute to true on that type.
|
// COM, set the ComVisible attribute to true on that type.
|
||||||
[assembly: ComVisible(false)]
|
[assembly: ComVisible(false)]
|
||||||
|
[assembly: InternalsVisibleTo("SafeExamBrowser.Applications.UnitTests")]
|
||||||
|
|
||||||
// The following GUID is for the ID of the typelib if this project is exposed to COM
|
// The following GUID is for the ID of the typelib if this project is exposed to COM
|
||||||
[assembly: Guid("a113e68f-1209-4689-981a-15c554b2df4e")]
|
[assembly: Guid("a113e68f-1209-4689-981a-15c554b2df4e")]
|
||||||
|
|
|
@ -9,9 +9,10 @@
|
||||||
<AppDesignerFolder>Properties</AppDesignerFolder>
|
<AppDesignerFolder>Properties</AppDesignerFolder>
|
||||||
<RootNamespace>SafeExamBrowser.Applications</RootNamespace>
|
<RootNamespace>SafeExamBrowser.Applications</RootNamespace>
|
||||||
<AssemblyName>SafeExamBrowser.Applications</AssemblyName>
|
<AssemblyName>SafeExamBrowser.Applications</AssemblyName>
|
||||||
<TargetFrameworkVersion>v4.7.2</TargetFrameworkVersion>
|
<TargetFrameworkVersion>v4.8</TargetFrameworkVersion>
|
||||||
<FileAlignment>512</FileAlignment>
|
<FileAlignment>512</FileAlignment>
|
||||||
<Deterministic>true</Deterministic>
|
<Deterministic>true</Deterministic>
|
||||||
|
<TargetFrameworkProfile />
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
<PropertyGroup Condition="'$(Configuration)|$(Platform)' == 'Debug|x86'">
|
<PropertyGroup Condition="'$(Configuration)|$(Platform)' == 'Debug|x86'">
|
||||||
<DebugSymbols>true</DebugSymbols>
|
<DebugSymbols>true</DebugSymbols>
|
||||||
|
@ -82,6 +83,10 @@
|
||||||
<Project>{30b2d907-5861-4f39-abad-c4abf1b3470e}</Project>
|
<Project>{30b2d907-5861-4f39-abad-c4abf1b3470e}</Project>
|
||||||
<Name>SafeExamBrowser.Settings</Name>
|
<Name>SafeExamBrowser.Settings</Name>
|
||||||
</ProjectReference>
|
</ProjectReference>
|
||||||
|
<ProjectReference Include="..\SafeExamBrowser.SystemComponents.Contracts\SafeExamBrowser.SystemComponents.Contracts.csproj">
|
||||||
|
<Project>{903129c6-e236-493b-9ad6-c6a57f647a3a}</Project>
|
||||||
|
<Name>SafeExamBrowser.SystemComponents.Contracts</Name>
|
||||||
|
</ProjectReference>
|
||||||
<ProjectReference Include="..\SafeExamBrowser.WindowsApi.Contracts\SafeExamBrowser.WindowsApi.Contracts.csproj">
|
<ProjectReference Include="..\SafeExamBrowser.WindowsApi.Contracts\SafeExamBrowser.WindowsApi.Contracts.csproj">
|
||||||
<Project>{7016f080-9aa5-41b2-a225-385ad877c171}</Project>
|
<Project>{7016f080-9aa5-41b2-a225-385ad877c171}</Project>
|
||||||
<Name>SafeExamBrowser.WindowsApi.Contracts</Name>
|
<Name>SafeExamBrowser.WindowsApi.Contracts</Name>
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
@ -9,7 +9,7 @@
|
||||||
namespace SafeExamBrowser.Browser.Contracts.Events
|
namespace SafeExamBrowser.Browser.Contracts.Events
|
||||||
{
|
{
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Event handler used to indicate that the browser has detected a session identifier of a LMS.
|
/// Event handler used to indicate that the browser has detected a user identifier of an LMS.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public delegate void SessionIdentifierDetectedEventHandler(string identifier);
|
public delegate void UserIdentifierDetectedEventHandler(string identifier);
|
||||||
}
|
}
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
@ -14,7 +14,7 @@ namespace SafeExamBrowser.Browser.Contracts
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Controls the lifetime and functionality of the browser application.
|
/// Controls the lifetime and functionality of the browser application.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public interface IBrowserApplication : IApplication
|
public interface IBrowserApplication : IApplication<IBrowserWindow>
|
||||||
{
|
{
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Event fired when the browser application detects a download request for an application configuration file.
|
/// Event fired when the browser application detects a download request for an application configuration file.
|
||||||
|
@ -22,9 +22,9 @@ namespace SafeExamBrowser.Browser.Contracts
|
||||||
event DownloadRequestedEventHandler ConfigurationDownloadRequested;
|
event DownloadRequestedEventHandler ConfigurationDownloadRequested;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Event fired when the browser application detects a session identifier of an LMS.
|
/// Event fired when the user tries to focus the taskbar.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
event SessionIdentifierDetectedEventHandler SessionIdentifierDetected;
|
event LoseFocusRequestedEventHandler LoseFocusRequested;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Event fired when the browser application detects a request to terminate SEB.
|
/// Event fired when the browser application detects a request to terminate SEB.
|
||||||
|
@ -32,9 +32,9 @@ namespace SafeExamBrowser.Browser.Contracts
|
||||||
event TerminationRequestedEventHandler TerminationRequested;
|
event TerminationRequestedEventHandler TerminationRequested;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Event fired when the user tries to focus the taskbar.
|
/// Event fired when the browser application detects a user identifier of an LMS.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
event LoseFocusRequestedEventHandler LoseFocusRequested;
|
event UserIdentifierDetectedEventHandler UserIdentifierDetected;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Transfers the focus to the browser application. If the parameter is <c>true</c>, the first focusable element in the browser window
|
/// Transfers the focus to the browser application. If the parameter is <c>true</c>, the first focusable element in the browser window
|
||||||
|
|
28
SafeExamBrowser.Browser.Contracts/IBrowserWindow.cs
Normal file
28
SafeExamBrowser.Browser.Contracts/IBrowserWindow.cs
Normal file
|
@ -0,0 +1,28 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
|
*
|
||||||
|
* 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 SafeExamBrowser.Applications.Contracts;
|
||||||
|
|
||||||
|
namespace SafeExamBrowser.Browser.Contracts
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Defines a window of the <see cref="IBrowserApplication"/>.
|
||||||
|
/// </summary>
|
||||||
|
public interface IBrowserWindow : IApplicationWindow
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Indicates whether the window is the main browser window.
|
||||||
|
/// </summary>
|
||||||
|
bool IsMainWindow { get; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// The currently loaded URL, or <c>default(string)</c> in case no navigation has happened yet.
|
||||||
|
/// </summary>
|
||||||
|
string Url { get; }
|
||||||
|
}
|
||||||
|
}
|
|
@ -8,7 +8,7 @@ using System.Runtime.InteropServices;
|
||||||
[assembly: AssemblyDescription("Safe Exam Browser")]
|
[assembly: AssemblyDescription("Safe Exam Browser")]
|
||||||
[assembly: AssemblyCompany("ETH Zürich")]
|
[assembly: AssemblyCompany("ETH Zürich")]
|
||||||
[assembly: AssemblyProduct("SafeExamBrowser.Browser.Contracts")]
|
[assembly: AssemblyProduct("SafeExamBrowser.Browser.Contracts")]
|
||||||
[assembly: AssemblyCopyright("Copyright © 2023 ETH Zürich, Educational Development and Technology (LET)")]
|
[assembly: AssemblyCopyright("Copyright © 2024 ETH Zürich, IT Services")]
|
||||||
|
|
||||||
// Setting ComVisible to false makes the types in this assembly not visible
|
// Setting ComVisible to false makes the types in this assembly not visible
|
||||||
// to COM components. If you need to access a type in this assembly from
|
// to COM components. If you need to access a type in this assembly from
|
||||||
|
|
|
@ -9,9 +9,10 @@
|
||||||
<AppDesignerFolder>Properties</AppDesignerFolder>
|
<AppDesignerFolder>Properties</AppDesignerFolder>
|
||||||
<RootNamespace>SafeExamBrowser.Browser.Contracts</RootNamespace>
|
<RootNamespace>SafeExamBrowser.Browser.Contracts</RootNamespace>
|
||||||
<AssemblyName>SafeExamBrowser.Browser.Contracts</AssemblyName>
|
<AssemblyName>SafeExamBrowser.Browser.Contracts</AssemblyName>
|
||||||
<TargetFrameworkVersion>v4.7.2</TargetFrameworkVersion>
|
<TargetFrameworkVersion>v4.8</TargetFrameworkVersion>
|
||||||
<FileAlignment>512</FileAlignment>
|
<FileAlignment>512</FileAlignment>
|
||||||
<Deterministic>true</Deterministic>
|
<Deterministic>true</Deterministic>
|
||||||
|
<TargetFrameworkProfile />
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
<PropertyGroup Condition="'$(Configuration)|$(Platform)' == 'Debug|x86'">
|
<PropertyGroup Condition="'$(Configuration)|$(Platform)' == 'Debug|x86'">
|
||||||
<DebugSymbols>true</DebugSymbols>
|
<DebugSymbols>true</DebugSymbols>
|
||||||
|
@ -59,13 +60,14 @@
|
||||||
<Compile Include="Events\DownloadRequestedEventHandler.cs" />
|
<Compile Include="Events\DownloadRequestedEventHandler.cs" />
|
||||||
<Compile Include="Events\TabPressedEventHandler.cs" />
|
<Compile Include="Events\TabPressedEventHandler.cs" />
|
||||||
<Compile Include="Events\LoseFocusRequestedEventHandler.cs" />
|
<Compile Include="Events\LoseFocusRequestedEventHandler.cs" />
|
||||||
<Compile Include="Events\SessionIdentifierDetectedEventHandler.cs" />
|
<Compile Include="Events\UserIdentifierDetectedEventHandler.cs" />
|
||||||
<Compile Include="Events\TerminationRequestedEventHandler.cs" />
|
<Compile Include="Events\TerminationRequestedEventHandler.cs" />
|
||||||
<Compile Include="Filters\IRequestFilter.cs" />
|
<Compile Include="Filters\IRequestFilter.cs" />
|
||||||
<Compile Include="Filters\IRule.cs" />
|
<Compile Include="Filters\IRule.cs" />
|
||||||
<Compile Include="Filters\IRuleFactory.cs" />
|
<Compile Include="Filters\IRuleFactory.cs" />
|
||||||
<Compile Include="Filters\Request.cs" />
|
<Compile Include="Filters\Request.cs" />
|
||||||
<Compile Include="IBrowserApplication.cs" />
|
<Compile Include="IBrowserApplication.cs" />
|
||||||
|
<Compile Include="IBrowserWindow.cs" />
|
||||||
<Compile Include="Properties\AssemblyInfo.cs" />
|
<Compile Include="Properties\AssemblyInfo.cs" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla internal
|
* This Source Code Form is subject to the terms of the Mozilla internal
|
||||||
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
@ -69,6 +69,7 @@ namespace SafeExamBrowser.Browser.UnitTests.Handlers
|
||||||
var quitUrl = "http://www.byebye.com";
|
var quitUrl = "http://www.byebye.com";
|
||||||
var request = new Mock<IRequest>();
|
var request = new Mock<IRequest>();
|
||||||
|
|
||||||
|
appConfig.ConfigurationFileMimeType = "application/seb";
|
||||||
request.SetupGet(r => r.Url).Returns(quitUrl);
|
request.SetupGet(r => r.Url).Returns(quitUrl);
|
||||||
settings.QuitUrl = quitUrl;
|
settings.QuitUrl = quitUrl;
|
||||||
sut.QuitUrlVisited += (url) => eventFired = true;
|
sut.QuitUrlVisited += (url) => eventFired = true;
|
||||||
|
@ -122,6 +123,7 @@ namespace SafeExamBrowser.Browser.UnitTests.Handlers
|
||||||
var request = new Mock<IRequest>();
|
var request = new Mock<IRequest>();
|
||||||
var url = "https://www.test.org";
|
var url = "https://www.test.org";
|
||||||
|
|
||||||
|
appConfig.ConfigurationFileMimeType = "application/seb";
|
||||||
filter.Setup(f => f.Process(It.Is<Request>(r => r.Url.Equals(url)))).Returns(FilterResult.Block);
|
filter.Setup(f => f.Process(It.Is<Request>(r => r.Url.Equals(url)))).Returns(FilterResult.Block);
|
||||||
request.SetupGet(r => r.ResourceType).Returns(ResourceType.MainFrame);
|
request.SetupGet(r => r.ResourceType).Returns(ResourceType.MainFrame);
|
||||||
request.SetupGet(r => r.Url).Returns(url);
|
request.SetupGet(r => r.Url).Returns(url);
|
||||||
|
@ -155,6 +157,7 @@ namespace SafeExamBrowser.Browser.UnitTests.Handlers
|
||||||
var request = new Mock<IRequest>();
|
var request = new Mock<IRequest>();
|
||||||
var url = "https://www.test.org";
|
var url = "https://www.test.org";
|
||||||
|
|
||||||
|
appConfig.ConfigurationFileMimeType = "application/seb";
|
||||||
filter.Setup(f => f.Process(It.Is<Request>(r => r.Url.Equals(url)))).Returns(FilterResult.Block);
|
filter.Setup(f => f.Process(It.Is<Request>(r => r.Url.Equals(url)))).Returns(FilterResult.Block);
|
||||||
request.SetupGet(r => r.ResourceType).Returns(ResourceType.SubFrame);
|
request.SetupGet(r => r.ResourceType).Returns(ResourceType.SubFrame);
|
||||||
request.SetupGet(r => r.Url).Returns(url);
|
request.SetupGet(r => r.Url).Returns(url);
|
||||||
|
@ -182,13 +185,34 @@ namespace SafeExamBrowser.Browser.UnitTests.Handlers
|
||||||
}
|
}
|
||||||
|
|
||||||
[TestMethod]
|
[TestMethod]
|
||||||
public void MustInitiateConfigurationFileDownload()
|
public void MustInitiateDataUriConfigurationFileDownload()
|
||||||
{
|
{
|
||||||
var browser = new Mock<IBrowser>();
|
var browser = new Mock<IBrowser>();
|
||||||
var host = new Mock<IBrowserHost>();
|
var host = new Mock<IBrowserHost>();
|
||||||
var request = new Mock<IRequest>();
|
var request = new Mock<IRequest>();
|
||||||
|
|
||||||
appConfig.ConfigurationFileExtension = ".xyz";
|
appConfig.ConfigurationFileExtension = ".xyz";
|
||||||
|
appConfig.ConfigurationFileMimeType = "application/seb";
|
||||||
|
appConfig.SebUriScheme = "abc";
|
||||||
|
appConfig.SebUriSchemeSecure = "abcd";
|
||||||
|
browser.Setup(b => b.GetHost()).Returns(host.Object);
|
||||||
|
request.SetupGet(r => r.Url).Returns($"{appConfig.SebUriSchemeSecure}://{appConfig.ConfigurationFileMimeType};base64,H4sIAAAAAAAAE41WbXPaRhD...");
|
||||||
|
|
||||||
|
var handled = sut.OnBeforeBrowse(Mock.Of<IWebBrowser>(), browser.Object, Mock.Of<IFrame>(), request.Object, false, false);
|
||||||
|
|
||||||
|
host.Verify(h => h.StartDownload(It.Is<string>(u => u == $"data:{appConfig.ConfigurationFileMimeType};base64,H4sIAAAAAAAAE41WbXPaRhD...")));
|
||||||
|
Assert.IsTrue(handled);
|
||||||
|
}
|
||||||
|
|
||||||
|
[TestMethod]
|
||||||
|
public void MustInitiateHttpConfigurationFileDownload()
|
||||||
|
{
|
||||||
|
var browser = new Mock<IBrowser>();
|
||||||
|
var host = new Mock<IBrowserHost>();
|
||||||
|
var request = new Mock<IRequest>();
|
||||||
|
|
||||||
|
appConfig.ConfigurationFileExtension = ".xyz";
|
||||||
|
appConfig.ConfigurationFileMimeType = "application/seb";
|
||||||
appConfig.SebUriScheme = "abc";
|
appConfig.SebUriScheme = "abc";
|
||||||
appConfig.SebUriSchemeSecure = "abcd";
|
appConfig.SebUriSchemeSecure = "abcd";
|
||||||
browser.Setup(b => b.GetHost()).Returns(host.Object);
|
browser.Setup(b => b.GetHost()).Returns(host.Object);
|
||||||
|
@ -198,13 +222,23 @@ namespace SafeExamBrowser.Browser.UnitTests.Handlers
|
||||||
|
|
||||||
host.Verify(h => h.StartDownload(It.Is<string>(u => u == $"{Uri.UriSchemeHttp}://host.com/path/file{appConfig.ConfigurationFileExtension}")));
|
host.Verify(h => h.StartDownload(It.Is<string>(u => u == $"{Uri.UriSchemeHttp}://host.com/path/file{appConfig.ConfigurationFileExtension}")));
|
||||||
Assert.IsTrue(handled);
|
Assert.IsTrue(handled);
|
||||||
|
}
|
||||||
|
|
||||||
handled = false;
|
[TestMethod]
|
||||||
host.Reset();
|
public void MustInitiateHttpsConfigurationFileDownload()
|
||||||
request.Reset();
|
{
|
||||||
|
var browser = new Mock<IBrowser>();
|
||||||
|
var host = new Mock<IBrowserHost>();
|
||||||
|
var request = new Mock<IRequest>();
|
||||||
|
|
||||||
|
appConfig.ConfigurationFileExtension = ".xyz";
|
||||||
|
appConfig.ConfigurationFileMimeType = "application/seb";
|
||||||
|
appConfig.SebUriScheme = "abc";
|
||||||
|
appConfig.SebUriSchemeSecure = "abcd";
|
||||||
|
browser.Setup(b => b.GetHost()).Returns(host.Object);
|
||||||
request.SetupGet(r => r.Url).Returns($"{appConfig.SebUriSchemeSecure}://host.com/path/file{appConfig.ConfigurationFileExtension}");
|
request.SetupGet(r => r.Url).Returns($"{appConfig.SebUriSchemeSecure}://host.com/path/file{appConfig.ConfigurationFileExtension}");
|
||||||
|
|
||||||
handled = sut.OnBeforeBrowse(Mock.Of<IWebBrowser>(), browser.Object, Mock.Of<IFrame>(), request.Object, false, false);
|
var handled = sut.OnBeforeBrowse(Mock.Of<IWebBrowser>(), browser.Object, Mock.Of<IFrame>(), request.Object, false, false);
|
||||||
|
|
||||||
host.Verify(h => h.StartDownload(It.Is<string>(u => u == $"{Uri.UriSchemeHttps}://host.com/path/file{appConfig.ConfigurationFileExtension}")));
|
host.Verify(h => h.StartDownload(It.Is<string>(u => u == $"{Uri.UriSchemeHttps}://host.com/path/file{appConfig.ConfigurationFileExtension}")));
|
||||||
Assert.IsTrue(handled);
|
Assert.IsTrue(handled);
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
@ -80,13 +80,41 @@ namespace SafeExamBrowser.Browser.UnitTests.Handlers
|
||||||
}
|
}
|
||||||
|
|
||||||
[TestMethod]
|
[TestMethod]
|
||||||
public void MustNotAppendCustomHeadersForCrossDomain()
|
public void MustAppendCustomHeadersForCrossDomainResourceRequestAndMainFrame()
|
||||||
{
|
{
|
||||||
var browser = new Mock<IWebBrowser>();
|
var browser = new Mock<IWebBrowser>();
|
||||||
var headers = new NameValueCollection();
|
var headers = new NameValueCollection();
|
||||||
var request = new Mock<IRequest>();
|
var request = new Mock<IRequest>();
|
||||||
|
|
||||||
browser.SetupGet(b => b.Address).Returns("http://www.otherhost.org");
|
browser.SetupGet(b => b.Address).Returns("http://www.otherhost.org");
|
||||||
|
keyGenerator.Setup(g => g.CalculateBrowserExamKeyHash(It.IsAny<string>(), It.IsAny<byte[]>(), It.IsAny<string>())).Returns(new Random().Next().ToString());
|
||||||
|
keyGenerator.Setup(g => g.CalculateConfigurationKeyHash(It.IsAny<string>(), It.IsAny<string>())).Returns(new Random().Next().ToString());
|
||||||
|
request.SetupGet(r => r.ResourceType).Returns(ResourceType.MainFrame);
|
||||||
|
request.SetupGet(r => r.Headers).Returns(new NameValueCollection());
|
||||||
|
request.SetupGet(r => r.Url).Returns("http://www.host.org");
|
||||||
|
request.SetupSet(r => r.Headers = It.IsAny<NameValueCollection>()).Callback<NameValueCollection>((h) => headers = h);
|
||||||
|
settings.SendConfigurationKey = true;
|
||||||
|
settings.SendBrowserExamKey = true;
|
||||||
|
|
||||||
|
var result = sut.OnBeforeResourceLoad(browser.Object, Mock.Of<IBrowser>(), Mock.Of<IFrame>(), request.Object, Mock.Of<IRequestCallback>());
|
||||||
|
|
||||||
|
request.VerifyGet(r => r.Headers, Times.AtLeastOnce);
|
||||||
|
request.VerifySet(r => r.Headers = It.IsAny<NameValueCollection>(), Times.AtLeastOnce);
|
||||||
|
|
||||||
|
Assert.AreEqual(CefReturnValue.Continue, result);
|
||||||
|
Assert.IsNotNull(headers["X-SafeExamBrowser-ConfigKeyHash"]);
|
||||||
|
Assert.IsNotNull(headers["X-SafeExamBrowser-RequestHash"]);
|
||||||
|
}
|
||||||
|
|
||||||
|
[TestMethod]
|
||||||
|
public void MustNotAppendCustomHeadersForCrossDomainResourceRequestAndSubResource()
|
||||||
|
{
|
||||||
|
var browser = new Mock<IWebBrowser>();
|
||||||
|
var headers = new NameValueCollection();
|
||||||
|
var request = new Mock<IRequest>();
|
||||||
|
|
||||||
|
browser.SetupGet(b => b.Address).Returns("http://www.otherhost.org");
|
||||||
|
request.SetupGet(r => r.ResourceType).Returns(ResourceType.SubResource);
|
||||||
request.SetupGet(r => r.Headers).Returns(new NameValueCollection());
|
request.SetupGet(r => r.Headers).Returns(new NameValueCollection());
|
||||||
request.SetupGet(r => r.Url).Returns("http://www.host.org");
|
request.SetupGet(r => r.Url).Returns("http://www.host.org");
|
||||||
request.SetupSet(r => r.Headers = It.IsAny<NameValueCollection>()).Callback<NameValueCollection>((h) => headers = h);
|
request.SetupSet(r => r.Headers = It.IsAny<NameValueCollection>()).Callback<NameValueCollection>((h) => headers = h);
|
||||||
|
@ -138,7 +166,7 @@ namespace SafeExamBrowser.Browser.UnitTests.Handlers
|
||||||
[TestMethod]
|
[TestMethod]
|
||||||
public void MustLetOperatingSystemHandleUnknownProtocols()
|
public void MustLetOperatingSystemHandleUnknownProtocols()
|
||||||
{
|
{
|
||||||
Assert.IsTrue(sut.OnProtocolExecution(default(IWebBrowser), default(IBrowser), default(IFrame), default(IRequest)));
|
Assert.IsTrue(sut.OnProtocolExecution(default, default, default, default));
|
||||||
}
|
}
|
||||||
|
|
||||||
[TestMethod]
|
[TestMethod]
|
||||||
|
@ -204,28 +232,28 @@ namespace SafeExamBrowser.Browser.UnitTests.Handlers
|
||||||
var newUrl = default(string);
|
var newUrl = default(string);
|
||||||
var request = new Mock<IRequest>();
|
var request = new Mock<IRequest>();
|
||||||
var response = new Mock<IResponse>();
|
var response = new Mock<IResponse>();
|
||||||
var sessionId = default(string);
|
var userId = default(string);
|
||||||
|
|
||||||
headers.Add("X-LMS-USER-ID", "some-session-id-123");
|
headers.Add("X-LMS-USER-ID", "some-session-id-123");
|
||||||
request.SetupGet(r => r.Url).Returns("https://www.somelms.org");
|
request.SetupGet(r => r.Url).Returns("https://www.somelms.org");
|
||||||
response.SetupGet(r => r.Headers).Returns(headers);
|
response.SetupGet(r => r.Headers).Returns(headers);
|
||||||
sut.SessionIdentifierDetected += (id) =>
|
sut.UserIdentifierDetected += (id) =>
|
||||||
{
|
{
|
||||||
sessionId = id;
|
userId = id;
|
||||||
@event.Set();
|
@event.Set();
|
||||||
};
|
};
|
||||||
|
|
||||||
sut.OnResourceRedirect(Mock.Of<IWebBrowser>(), Mock.Of<IBrowser>(), Mock.Of<IFrame>(), Mock.Of<IRequest>(), response.Object, ref newUrl);
|
sut.OnResourceRedirect(Mock.Of<IWebBrowser>(), Mock.Of<IBrowser>(), Mock.Of<IFrame>(), Mock.Of<IRequest>(), response.Object, ref newUrl);
|
||||||
@event.WaitOne();
|
@event.WaitOne();
|
||||||
Assert.AreEqual("some-session-id-123", sessionId);
|
Assert.AreEqual("some-session-id-123", userId);
|
||||||
|
|
||||||
headers.Clear();
|
headers.Clear();
|
||||||
headers.Add("X-LMS-USER-ID", "other-session-id-123");
|
headers.Add("X-LMS-USER-ID", "other-session-id-123");
|
||||||
sessionId = default(string);
|
userId = default;
|
||||||
|
|
||||||
sut.OnResourceResponse(Mock.Of<IWebBrowser>(), Mock.Of<IBrowser>(), Mock.Of<IFrame>(), request.Object, response.Object);
|
sut.OnResourceResponse(Mock.Of<IWebBrowser>(), Mock.Of<IBrowser>(), Mock.Of<IFrame>(), request.Object, response.Object);
|
||||||
@event.WaitOne();
|
@event.WaitOne();
|
||||||
Assert.AreEqual("other-session-id-123", sessionId);
|
Assert.AreEqual("other-session-id-123", userId);
|
||||||
}
|
}
|
||||||
|
|
||||||
[TestMethod]
|
[TestMethod]
|
||||||
|
@ -236,28 +264,28 @@ namespace SafeExamBrowser.Browser.UnitTests.Handlers
|
||||||
var newUrl = default(string);
|
var newUrl = default(string);
|
||||||
var request = new Mock<IRequest>();
|
var request = new Mock<IRequest>();
|
||||||
var response = new Mock<IResponse>();
|
var response = new Mock<IResponse>();
|
||||||
var sessionId = default(string);
|
var userId = default(string);
|
||||||
|
|
||||||
headers.Add("Set-Cookie", "edx-user-info=\"{\\\"username\\\": \\\"edx-123\\\"}\"; expires");
|
headers.Add("Set-Cookie", "edx-user-info=\"{\\\"username\\\": \\\"edx-123\\\"}\"; expires");
|
||||||
request.SetupGet(r => r.Url).Returns("https://www.somelms.org");
|
request.SetupGet(r => r.Url).Returns("https://www.somelms.org");
|
||||||
response.SetupGet(r => r.Headers).Returns(headers);
|
response.SetupGet(r => r.Headers).Returns(headers);
|
||||||
sut.SessionIdentifierDetected += (id) =>
|
sut.UserIdentifierDetected += (id) =>
|
||||||
{
|
{
|
||||||
sessionId = id;
|
userId = id;
|
||||||
@event.Set();
|
@event.Set();
|
||||||
};
|
};
|
||||||
|
|
||||||
sut.OnResourceRedirect(Mock.Of<IWebBrowser>(), Mock.Of<IBrowser>(), Mock.Of<IFrame>(), Mock.Of<IRequest>(), response.Object, ref newUrl);
|
sut.OnResourceRedirect(Mock.Of<IWebBrowser>(), Mock.Of<IBrowser>(), Mock.Of<IFrame>(), Mock.Of<IRequest>(), response.Object, ref newUrl);
|
||||||
@event.WaitOne();
|
@event.WaitOne();
|
||||||
Assert.AreEqual("edx-123", sessionId);
|
Assert.AreEqual("edx-123", userId);
|
||||||
|
|
||||||
headers.Clear();
|
headers.Clear();
|
||||||
headers.Add("Set-Cookie", "edx-user-info=\"{\\\"username\\\": \\\"edx-345\\\"}\"; expires");
|
headers.Add("Set-Cookie", "edx-user-info=\"{\\\"username\\\": \\\"edx-345\\\"}\"; expires");
|
||||||
sessionId = default(string);
|
userId = default;
|
||||||
|
|
||||||
sut.OnResourceResponse(Mock.Of<IWebBrowser>(), Mock.Of<IBrowser>(), Mock.Of<IFrame>(), request.Object, response.Object);
|
sut.OnResourceResponse(Mock.Of<IWebBrowser>(), Mock.Of<IBrowser>(), Mock.Of<IFrame>(), request.Object, response.Object);
|
||||||
@event.WaitOne();
|
@event.WaitOne();
|
||||||
Assert.AreEqual("edx-345", sessionId);
|
Assert.AreEqual("edx-345", userId);
|
||||||
}
|
}
|
||||||
|
|
||||||
[TestMethod]
|
[TestMethod]
|
||||||
|
@ -268,28 +296,28 @@ namespace SafeExamBrowser.Browser.UnitTests.Handlers
|
||||||
var newUrl = default(string);
|
var newUrl = default(string);
|
||||||
var request = new Mock<IRequest>();
|
var request = new Mock<IRequest>();
|
||||||
var response = new Mock<IResponse>();
|
var response = new Mock<IResponse>();
|
||||||
var sessionId = default(string);
|
var userId = default(string);
|
||||||
|
|
||||||
headers.Add("Location", "https://www.some-moodle-instance.org/moodle/login/index.php?testsession=123");
|
headers.Add("Location", "https://www.some-moodle-instance.org/moodle/login/index.php?testsession=123");
|
||||||
request.SetupGet(r => r.Url).Returns("https://www.some-moodle-instance.org");
|
request.SetupGet(r => r.Url).Returns("https://www.some-moodle-instance.org");
|
||||||
response.SetupGet(r => r.Headers).Returns(headers);
|
response.SetupGet(r => r.Headers).Returns(headers);
|
||||||
sut.SessionIdentifierDetected += (id) =>
|
sut.UserIdentifierDetected += (id) =>
|
||||||
{
|
{
|
||||||
sessionId = id;
|
userId = id;
|
||||||
@event.Set();
|
@event.Set();
|
||||||
};
|
};
|
||||||
|
|
||||||
sut.OnResourceRedirect(Mock.Of<IWebBrowser>(), Mock.Of<IBrowser>(), Mock.Of<IFrame>(), Mock.Of<IRequest>(), response.Object, ref newUrl);
|
sut.OnResourceRedirect(Mock.Of<IWebBrowser>(), Mock.Of<IBrowser>(), Mock.Of<IFrame>(), Mock.Of<IRequest>(), response.Object, ref newUrl);
|
||||||
@event.WaitOne();
|
@event.WaitOne();
|
||||||
Assert.AreEqual("123", sessionId);
|
Assert.AreEqual("123", userId);
|
||||||
|
|
||||||
headers.Clear();
|
headers.Clear();
|
||||||
headers.Add("Location", "https://www.some-moodle-instance.org/moodle/login/index.php?testsession=456");
|
headers.Add("Location", "https://www.some-moodle-instance.org/moodle/login/index.php?testsession=456");
|
||||||
sessionId = default(string);
|
userId = default;
|
||||||
|
|
||||||
sut.OnResourceResponse(Mock.Of<IWebBrowser>(), Mock.Of<IBrowser>(), Mock.Of<IFrame>(), request.Object, response.Object);
|
sut.OnResourceResponse(Mock.Of<IWebBrowser>(), Mock.Of<IBrowser>(), Mock.Of<IFrame>(), request.Object, response.Object);
|
||||||
@event.WaitOne();
|
@event.WaitOne();
|
||||||
Assert.AreEqual("456", sessionId);
|
Assert.AreEqual("456", userId);
|
||||||
}
|
}
|
||||||
|
|
||||||
private class TestableResourceHandler : ResourceHandler
|
private class TestableResourceHandler : ResourceHandler
|
||||||
|
|
|
@ -5,7 +5,7 @@ using System.Runtime.InteropServices;
|
||||||
[assembly: AssemblyDescription("Safe Exam Browser")]
|
[assembly: AssemblyDescription("Safe Exam Browser")]
|
||||||
[assembly: AssemblyCompany("ETH Zürich")]
|
[assembly: AssemblyCompany("ETH Zürich")]
|
||||||
[assembly: AssemblyProduct("SafeExamBrowser.Browser.UnitTests")]
|
[assembly: AssemblyProduct("SafeExamBrowser.Browser.UnitTests")]
|
||||||
[assembly: AssemblyCopyright("Copyright © 2023 ETH Zürich, Educational Development and Technology (LET)")]
|
[assembly: AssemblyCopyright("Copyright © 2024 ETH Zürich, IT Services")]
|
||||||
|
|
||||||
[assembly: ComVisible(false)]
|
[assembly: ComVisible(false)]
|
||||||
|
|
||||||
|
|
|
@ -1,9 +1,11 @@
|
||||||
<?xml version="1.0" encoding="utf-8"?>
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
<Project ToolsVersion="15.0" DefaultTargets="Build" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
|
<Project ToolsVersion="15.0" DefaultTargets="Build" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
|
||||||
<Import Project="..\packages\CefSharp.Common.111.2.20\build\CefSharp.Common.props" Condition="Exists('..\packages\CefSharp.Common.111.2.20\build\CefSharp.Common.props')" />
|
<Import Project="..\packages\MSTest.TestAdapter.3.2.2\build\net462\MSTest.TestAdapter.props" Condition="Exists('..\packages\MSTest.TestAdapter.3.2.2\build\net462\MSTest.TestAdapter.props')" />
|
||||||
<Import Project="..\packages\cef.redist.x86.111.2.2\build\cef.redist.x86.props" Condition="Exists('..\packages\cef.redist.x86.111.2.2\build\cef.redist.x86.props')" />
|
<Import Project="..\packages\Microsoft.Testing.Extensions.Telemetry.1.0.2\build\netstandard2.0\Microsoft.Testing.Extensions.Telemetry.props" Condition="Exists('..\packages\Microsoft.Testing.Extensions.Telemetry.1.0.2\build\netstandard2.0\Microsoft.Testing.Extensions.Telemetry.props')" />
|
||||||
<Import Project="..\packages\cef.redist.x64.111.2.2\build\cef.redist.x64.props" Condition="Exists('..\packages\cef.redist.x64.111.2.2\build\cef.redist.x64.props')" />
|
<Import Project="..\packages\Microsoft.Testing.Platform.MSBuild.1.0.2\build\netstandard2.0\Microsoft.Testing.Platform.MSBuild.props" Condition="Exists('..\packages\Microsoft.Testing.Platform.MSBuild.1.0.2\build\netstandard2.0\Microsoft.Testing.Platform.MSBuild.props')" />
|
||||||
<Import Project="..\packages\MSTest.TestAdapter.3.0.2\build\net462\MSTest.TestAdapter.props" Condition="Exists('..\packages\MSTest.TestAdapter.3.0.2\build\net462\MSTest.TestAdapter.props')" />
|
<Import Project="..\packages\CefSharp.Common.121.3.130\build\CefSharp.Common.props" Condition="Exists('..\packages\CefSharp.Common.121.3.130\build\CefSharp.Common.props')" />
|
||||||
|
<Import Project="..\packages\chromiumembeddedframework.runtime.win-x86.121.3.13\build\chromiumembeddedframework.runtime.win-x86.props" Condition="Exists('..\packages\chromiumembeddedframework.runtime.win-x86.121.3.13\build\chromiumembeddedframework.runtime.win-x86.props')" />
|
||||||
|
<Import Project="..\packages\chromiumembeddedframework.runtime.win-x64.121.3.13\build\chromiumembeddedframework.runtime.win-x64.props" Condition="Exists('..\packages\chromiumembeddedframework.runtime.win-x64.121.3.13\build\chromiumembeddedframework.runtime.win-x64.props')" />
|
||||||
<Import Project="$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props" Condition="Exists('$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props')" />
|
<Import Project="$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props" Condition="Exists('$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props')" />
|
||||||
<PropertyGroup>
|
<PropertyGroup>
|
||||||
<Configuration Condition=" '$(Configuration)' == '' ">Debug</Configuration>
|
<Configuration Condition=" '$(Configuration)' == '' ">Debug</Configuration>
|
||||||
|
@ -13,7 +15,7 @@
|
||||||
<AppDesignerFolder>Properties</AppDesignerFolder>
|
<AppDesignerFolder>Properties</AppDesignerFolder>
|
||||||
<RootNamespace>SafeExamBrowser.Browser.UnitTests</RootNamespace>
|
<RootNamespace>SafeExamBrowser.Browser.UnitTests</RootNamespace>
|
||||||
<AssemblyName>SafeExamBrowser.Browser.UnitTests</AssemblyName>
|
<AssemblyName>SafeExamBrowser.Browser.UnitTests</AssemblyName>
|
||||||
<TargetFrameworkVersion>v4.7.2</TargetFrameworkVersion>
|
<TargetFrameworkVersion>v4.8</TargetFrameworkVersion>
|
||||||
<FileAlignment>512</FileAlignment>
|
<FileAlignment>512</FileAlignment>
|
||||||
<ProjectTypeGuids>{3AC096D0-A1C2-E12C-1390-A8335801FDAB};{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}</ProjectTypeGuids>
|
<ProjectTypeGuids>{3AC096D0-A1C2-E12C-1390-A8335801FDAB};{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}</ProjectTypeGuids>
|
||||||
<VisualStudioVersion Condition="'$(VisualStudioVersion)' == ''">15.0</VisualStudioVersion>
|
<VisualStudioVersion Condition="'$(VisualStudioVersion)' == ''">15.0</VisualStudioVersion>
|
||||||
|
@ -23,6 +25,7 @@
|
||||||
<TestProjectType>UnitTest</TestProjectType>
|
<TestProjectType>UnitTest</TestProjectType>
|
||||||
<NuGetPackageImportStamp>
|
<NuGetPackageImportStamp>
|
||||||
</NuGetPackageImportStamp>
|
</NuGetPackageImportStamp>
|
||||||
|
<TargetFrameworkProfile />
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
<PropertyGroup Condition="'$(Configuration)|$(Platform)' == 'Debug|x86'">
|
<PropertyGroup Condition="'$(Configuration)|$(Platform)' == 'Debug|x86'">
|
||||||
<DebugSymbols>true</DebugSymbols>
|
<DebugSymbols>true</DebugSymbols>
|
||||||
|
@ -64,31 +67,85 @@
|
||||||
<Reference Include="Castle.Core, Version=5.0.0.0, Culture=neutral, PublicKeyToken=407dd0808d44fbdc, processorArchitecture=MSIL">
|
<Reference Include="Castle.Core, Version=5.0.0.0, Culture=neutral, PublicKeyToken=407dd0808d44fbdc, processorArchitecture=MSIL">
|
||||||
<HintPath>..\packages\Castle.Core.5.1.1\lib\net462\Castle.Core.dll</HintPath>
|
<HintPath>..\packages\Castle.Core.5.1.1\lib\net462\Castle.Core.dll</HintPath>
|
||||||
</Reference>
|
</Reference>
|
||||||
<Reference Include="CefSharp, Version=111.2.20.0, Culture=neutral, PublicKeyToken=40c4b6fc221f4138, processorArchitecture=MSIL">
|
<Reference Include="CefSharp, Version=121.3.130.0, Culture=neutral, PublicKeyToken=40c4b6fc221f4138, processorArchitecture=MSIL">
|
||||||
<HintPath>..\packages\CefSharp.Common.111.2.20\lib\net452\CefSharp.dll</HintPath>
|
<HintPath>..\packages\CefSharp.Common.121.3.130\lib\net462\CefSharp.dll</HintPath>
|
||||||
</Reference>
|
</Reference>
|
||||||
<Reference Include="CefSharp.Core, Version=111.2.20.0, Culture=neutral, PublicKeyToken=40c4b6fc221f4138, processorArchitecture=MSIL">
|
<Reference Include="CefSharp.Core, Version=121.3.130.0, Culture=neutral, PublicKeyToken=40c4b6fc221f4138, processorArchitecture=MSIL">
|
||||||
<HintPath>..\packages\CefSharp.Common.111.2.20\lib\net452\CefSharp.Core.dll</HintPath>
|
<HintPath>..\packages\CefSharp.Common.121.3.130\lib\net462\CefSharp.Core.dll</HintPath>
|
||||||
|
</Reference>
|
||||||
|
<Reference Include="Microsoft.ApplicationInsights, Version=2.22.0.997, Culture=neutral, PublicKeyToken=31bf3856ad364e35, processorArchitecture=MSIL">
|
||||||
|
<HintPath>..\packages\Microsoft.ApplicationInsights.2.22.0\lib\net46\Microsoft.ApplicationInsights.dll</HintPath>
|
||||||
|
</Reference>
|
||||||
|
<Reference Include="Microsoft.CSharp" />
|
||||||
|
<Reference Include="Microsoft.Testing.Extensions.Telemetry, Version=1.0.2.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL">
|
||||||
|
<HintPath>..\packages\Microsoft.Testing.Extensions.Telemetry.1.0.2\lib\netstandard2.0\Microsoft.Testing.Extensions.Telemetry.dll</HintPath>
|
||||||
|
</Reference>
|
||||||
|
<Reference Include="Microsoft.Testing.Extensions.TrxReport.Abstractions, Version=1.0.2.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL">
|
||||||
|
<HintPath>..\packages\Microsoft.Testing.Extensions.TrxReport.Abstractions.1.0.2\lib\netstandard2.0\Microsoft.Testing.Extensions.TrxReport.Abstractions.dll</HintPath>
|
||||||
|
</Reference>
|
||||||
|
<Reference Include="Microsoft.Testing.Extensions.VSTestBridge, Version=1.0.2.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL">
|
||||||
|
<HintPath>..\packages\Microsoft.Testing.Extensions.VSTestBridge.1.0.2\lib\netstandard2.0\Microsoft.Testing.Extensions.VSTestBridge.dll</HintPath>
|
||||||
|
</Reference>
|
||||||
|
<Reference Include="Microsoft.Testing.Platform, Version=1.0.2.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL">
|
||||||
|
<HintPath>..\packages\Microsoft.Testing.Platform.MSBuild.1.0.2\lib\netstandard2.0\Microsoft.Testing.Platform.dll</HintPath>
|
||||||
|
</Reference>
|
||||||
|
<Reference Include="Microsoft.Testing.Platform.MSBuild, Version=1.0.2.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL">
|
||||||
|
<HintPath>..\packages\Microsoft.Testing.Platform.MSBuild.1.0.2\lib\netstandard2.0\Microsoft.Testing.Platform.MSBuild.dll</HintPath>
|
||||||
|
</Reference>
|
||||||
|
<Reference Include="Microsoft.TestPlatform.CoreUtilities, Version=15.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL">
|
||||||
|
<HintPath>..\packages\Microsoft.TestPlatform.ObjectModel.17.9.0\lib\net462\Microsoft.TestPlatform.CoreUtilities.dll</HintPath>
|
||||||
|
</Reference>
|
||||||
|
<Reference Include="Microsoft.TestPlatform.PlatformAbstractions, Version=15.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL">
|
||||||
|
<HintPath>..\packages\Microsoft.TestPlatform.ObjectModel.17.9.0\lib\net462\Microsoft.TestPlatform.PlatformAbstractions.dll</HintPath>
|
||||||
|
</Reference>
|
||||||
|
<Reference Include="Microsoft.VisualStudio.TestPlatform.ObjectModel, Version=15.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL">
|
||||||
|
<HintPath>..\packages\Microsoft.TestPlatform.ObjectModel.17.9.0\lib\net462\Microsoft.VisualStudio.TestPlatform.ObjectModel.dll</HintPath>
|
||||||
</Reference>
|
</Reference>
|
||||||
<Reference Include="Microsoft.VisualStudio.TestPlatform.TestFramework, Version=14.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL">
|
<Reference Include="Microsoft.VisualStudio.TestPlatform.TestFramework, Version=14.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL">
|
||||||
<HintPath>..\packages\MSTest.TestFramework.3.0.2\lib\net462\Microsoft.VisualStudio.TestPlatform.TestFramework.dll</HintPath>
|
<HintPath>..\packages\MSTest.TestFramework.3.2.2\lib\net462\Microsoft.VisualStudio.TestPlatform.TestFramework.dll</HintPath>
|
||||||
</Reference>
|
</Reference>
|
||||||
<Reference Include="Microsoft.VisualStudio.TestPlatform.TestFramework.Extensions, Version=14.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL">
|
<Reference Include="Microsoft.VisualStudio.TestPlatform.TestFramework.Extensions, Version=14.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL">
|
||||||
<HintPath>..\packages\MSTest.TestFramework.3.0.2\lib\net462\Microsoft.VisualStudio.TestPlatform.TestFramework.Extensions.dll</HintPath>
|
<HintPath>..\packages\MSTest.TestFramework.3.2.2\lib\net462\Microsoft.VisualStudio.TestPlatform.TestFramework.Extensions.dll</HintPath>
|
||||||
</Reference>
|
</Reference>
|
||||||
<Reference Include="Moq, Version=4.18.0.0, Culture=neutral, PublicKeyToken=69f491c39445e920, processorArchitecture=MSIL">
|
<Reference Include="Moq, Version=4.20.70.0, Culture=neutral, PublicKeyToken=69f491c39445e920, processorArchitecture=MSIL">
|
||||||
<HintPath>..\packages\Moq.4.18.4\lib\net462\Moq.dll</HintPath>
|
<HintPath>..\packages\Moq.4.20.70\lib\net462\Moq.dll</HintPath>
|
||||||
|
</Reference>
|
||||||
|
<Reference Include="NuGet.Frameworks, Version=6.9.1.3, Culture=neutral, PublicKeyToken=31bf3856ad364e35, processorArchitecture=MSIL">
|
||||||
|
<HintPath>..\packages\NuGet.Frameworks.6.9.1\lib\net472\NuGet.Frameworks.dll</HintPath>
|
||||||
</Reference>
|
</Reference>
|
||||||
<Reference Include="System" />
|
<Reference Include="System" />
|
||||||
|
<Reference Include="System.Buffers, Version=4.0.3.0, Culture=neutral, PublicKeyToken=cc7b13ffcd2ddd51, processorArchitecture=MSIL">
|
||||||
|
<HintPath>..\packages\System.Buffers.4.5.1\lib\net461\System.Buffers.dll</HintPath>
|
||||||
|
</Reference>
|
||||||
|
<Reference Include="System.Collections.Immutable, Version=8.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL">
|
||||||
|
<HintPath>..\packages\System.Collections.Immutable.8.0.0\lib\net462\System.Collections.Immutable.dll</HintPath>
|
||||||
|
</Reference>
|
||||||
<Reference Include="System.Configuration" />
|
<Reference Include="System.Configuration" />
|
||||||
<Reference Include="System.Core" />
|
<Reference Include="System.Core" />
|
||||||
|
<Reference Include="System.Diagnostics.DiagnosticSource, Version=8.0.0.0, Culture=neutral, PublicKeyToken=cc7b13ffcd2ddd51, processorArchitecture=MSIL">
|
||||||
|
<HintPath>..\packages\System.Diagnostics.DiagnosticSource.8.0.0\lib\net462\System.Diagnostics.DiagnosticSource.dll</HintPath>
|
||||||
|
</Reference>
|
||||||
|
<Reference Include="System.Memory, Version=4.0.1.2, Culture=neutral, PublicKeyToken=cc7b13ffcd2ddd51, processorArchitecture=MSIL">
|
||||||
|
<HintPath>..\packages\System.Memory.4.5.5\lib\net461\System.Memory.dll</HintPath>
|
||||||
|
</Reference>
|
||||||
|
<Reference Include="System.Net.Http" />
|
||||||
|
<Reference Include="System.Numerics" />
|
||||||
|
<Reference Include="System.Numerics.Vectors, Version=4.1.4.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL">
|
||||||
|
<HintPath>..\packages\System.Numerics.Vectors.4.5.0\lib\net46\System.Numerics.Vectors.dll</HintPath>
|
||||||
|
</Reference>
|
||||||
|
<Reference Include="System.Reflection.Metadata, Version=8.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL">
|
||||||
|
<HintPath>..\packages\System.Reflection.Metadata.8.0.0\lib\net462\System.Reflection.Metadata.dll</HintPath>
|
||||||
|
</Reference>
|
||||||
|
<Reference Include="System.Runtime" />
|
||||||
<Reference Include="System.Runtime.CompilerServices.Unsafe, Version=6.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL">
|
<Reference Include="System.Runtime.CompilerServices.Unsafe, Version=6.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL">
|
||||||
<HintPath>..\packages\System.Runtime.CompilerServices.Unsafe.6.0.0\lib\net461\System.Runtime.CompilerServices.Unsafe.dll</HintPath>
|
<HintPath>..\packages\System.Runtime.CompilerServices.Unsafe.6.0.0\lib\net461\System.Runtime.CompilerServices.Unsafe.dll</HintPath>
|
||||||
</Reference>
|
</Reference>
|
||||||
|
<Reference Include="System.Runtime.Serialization" />
|
||||||
<Reference Include="System.Threading.Tasks.Extensions, Version=4.2.0.1, Culture=neutral, PublicKeyToken=cc7b13ffcd2ddd51, processorArchitecture=MSIL">
|
<Reference Include="System.Threading.Tasks.Extensions, Version=4.2.0.1, Culture=neutral, PublicKeyToken=cc7b13ffcd2ddd51, processorArchitecture=MSIL">
|
||||||
<HintPath>..\packages\System.Threading.Tasks.Extensions.4.5.4\lib\net461\System.Threading.Tasks.Extensions.dll</HintPath>
|
<HintPath>..\packages\System.Threading.Tasks.Extensions.4.5.4\lib\net461\System.Threading.Tasks.Extensions.dll</HintPath>
|
||||||
</Reference>
|
</Reference>
|
||||||
<Reference Include="System.Windows.Forms" />
|
<Reference Include="System.Windows.Forms" />
|
||||||
|
<Reference Include="System.Xml" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<Compile Include="Filters\LegacyFilter.cs" />
|
<Compile Include="Filters\LegacyFilter.cs" />
|
||||||
|
@ -109,9 +166,7 @@
|
||||||
<None Include="app.config">
|
<None Include="app.config">
|
||||||
<SubType>Designer</SubType>
|
<SubType>Designer</SubType>
|
||||||
</None>
|
</None>
|
||||||
<None Include="packages.config">
|
<None Include="packages.config" />
|
||||||
<SubType>Designer</SubType>
|
|
||||||
</None>
|
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<ProjectReference Include="..\SafeExamBrowser.Browser.Contracts\SafeExamBrowser.Browser.Contracts.csproj">
|
<ProjectReference Include="..\SafeExamBrowser.Browser.Contracts\SafeExamBrowser.Browser.Contracts.csproj">
|
||||||
|
@ -149,13 +204,17 @@
|
||||||
<PropertyGroup>
|
<PropertyGroup>
|
||||||
<ErrorText>This project references NuGet package(s) that are missing on this computer. Use NuGet Package Restore to download them. For more information, see http://go.microsoft.com/fwlink/?LinkID=322105. The missing file is {0}.</ErrorText>
|
<ErrorText>This project references NuGet package(s) that are missing on this computer. Use NuGet Package Restore to download them. For more information, see http://go.microsoft.com/fwlink/?LinkID=322105. The missing file is {0}.</ErrorText>
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
<Error Condition="!Exists('..\packages\MSTest.TestAdapter.3.0.2\build\net462\MSTest.TestAdapter.props')" Text="$([System.String]::Format('$(ErrorText)', '..\packages\MSTest.TestAdapter.3.0.2\build\net462\MSTest.TestAdapter.props'))" />
|
<Error Condition="!Exists('..\packages\chromiumembeddedframework.runtime.win-x64.121.3.13\build\chromiumembeddedframework.runtime.win-x64.props')" Text="$([System.String]::Format('$(ErrorText)', '..\packages\chromiumembeddedframework.runtime.win-x64.121.3.13\build\chromiumembeddedframework.runtime.win-x64.props'))" />
|
||||||
<Error Condition="!Exists('..\packages\MSTest.TestAdapter.3.0.2\build\net462\MSTest.TestAdapter.targets')" Text="$([System.String]::Format('$(ErrorText)', '..\packages\MSTest.TestAdapter.3.0.2\build\net462\MSTest.TestAdapter.targets'))" />
|
<Error Condition="!Exists('..\packages\chromiumembeddedframework.runtime.win-x86.121.3.13\build\chromiumembeddedframework.runtime.win-x86.props')" Text="$([System.String]::Format('$(ErrorText)', '..\packages\chromiumembeddedframework.runtime.win-x86.121.3.13\build\chromiumembeddedframework.runtime.win-x86.props'))" />
|
||||||
<Error Condition="!Exists('..\packages\cef.redist.x64.111.2.2\build\cef.redist.x64.props')" Text="$([System.String]::Format('$(ErrorText)', '..\packages\cef.redist.x64.111.2.2\build\cef.redist.x64.props'))" />
|
<Error Condition="!Exists('..\packages\CefSharp.Common.121.3.130\build\CefSharp.Common.props')" Text="$([System.String]::Format('$(ErrorText)', '..\packages\CefSharp.Common.121.3.130\build\CefSharp.Common.props'))" />
|
||||||
<Error Condition="!Exists('..\packages\cef.redist.x86.111.2.2\build\cef.redist.x86.props')" Text="$([System.String]::Format('$(ErrorText)', '..\packages\cef.redist.x86.111.2.2\build\cef.redist.x86.props'))" />
|
<Error Condition="!Exists('..\packages\CefSharp.Common.121.3.130\build\CefSharp.Common.targets')" Text="$([System.String]::Format('$(ErrorText)', '..\packages\CefSharp.Common.121.3.130\build\CefSharp.Common.targets'))" />
|
||||||
<Error Condition="!Exists('..\packages\CefSharp.Common.111.2.20\build\CefSharp.Common.props')" Text="$([System.String]::Format('$(ErrorText)', '..\packages\CefSharp.Common.111.2.20\build\CefSharp.Common.props'))" />
|
<Error Condition="!Exists('..\packages\Microsoft.Testing.Platform.MSBuild.1.0.2\build\netstandard2.0\Microsoft.Testing.Platform.MSBuild.props')" Text="$([System.String]::Format('$(ErrorText)', '..\packages\Microsoft.Testing.Platform.MSBuild.1.0.2\build\netstandard2.0\Microsoft.Testing.Platform.MSBuild.props'))" />
|
||||||
<Error Condition="!Exists('..\packages\CefSharp.Common.111.2.20\build\CefSharp.Common.targets')" Text="$([System.String]::Format('$(ErrorText)', '..\packages\CefSharp.Common.111.2.20\build\CefSharp.Common.targets'))" />
|
<Error Condition="!Exists('..\packages\Microsoft.Testing.Platform.MSBuild.1.0.2\build\netstandard2.0\Microsoft.Testing.Platform.MSBuild.targets')" Text="$([System.String]::Format('$(ErrorText)', '..\packages\Microsoft.Testing.Platform.MSBuild.1.0.2\build\netstandard2.0\Microsoft.Testing.Platform.MSBuild.targets'))" />
|
||||||
|
<Error Condition="!Exists('..\packages\Microsoft.Testing.Extensions.Telemetry.1.0.2\build\netstandard2.0\Microsoft.Testing.Extensions.Telemetry.props')" Text="$([System.String]::Format('$(ErrorText)', '..\packages\Microsoft.Testing.Extensions.Telemetry.1.0.2\build\netstandard2.0\Microsoft.Testing.Extensions.Telemetry.props'))" />
|
||||||
|
<Error Condition="!Exists('..\packages\MSTest.TestAdapter.3.2.2\build\net462\MSTest.TestAdapter.props')" Text="$([System.String]::Format('$(ErrorText)', '..\packages\MSTest.TestAdapter.3.2.2\build\net462\MSTest.TestAdapter.props'))" />
|
||||||
|
<Error Condition="!Exists('..\packages\MSTest.TestAdapter.3.2.2\build\net462\MSTest.TestAdapter.targets')" Text="$([System.String]::Format('$(ErrorText)', '..\packages\MSTest.TestAdapter.3.2.2\build\net462\MSTest.TestAdapter.targets'))" />
|
||||||
</Target>
|
</Target>
|
||||||
<Import Project="..\packages\MSTest.TestAdapter.3.0.2\build\net462\MSTest.TestAdapter.targets" Condition="Exists('..\packages\MSTest.TestAdapter.3.0.2\build\net462\MSTest.TestAdapter.targets')" />
|
<Import Project="..\packages\CefSharp.Common.121.3.130\build\CefSharp.Common.targets" Condition="Exists('..\packages\CefSharp.Common.121.3.130\build\CefSharp.Common.targets')" />
|
||||||
<Import Project="..\packages\CefSharp.Common.111.2.20\build\CefSharp.Common.targets" Condition="Exists('..\packages\CefSharp.Common.111.2.20\build\CefSharp.Common.targets')" />
|
<Import Project="..\packages\Microsoft.Testing.Platform.MSBuild.1.0.2\build\netstandard2.0\Microsoft.Testing.Platform.MSBuild.targets" Condition="Exists('..\packages\Microsoft.Testing.Platform.MSBuild.1.0.2\build\netstandard2.0\Microsoft.Testing.Platform.MSBuild.targets')" />
|
||||||
|
<Import Project="..\packages\MSTest.TestAdapter.3.2.2\build\net462\MSTest.TestAdapter.targets" Condition="Exists('..\packages\MSTest.TestAdapter.3.2.2\build\net462\MSTest.TestAdapter.targets')" />
|
||||||
</Project>
|
</Project>
|
|
@ -16,12 +16,36 @@
|
||||||
</dependentAssembly>
|
</dependentAssembly>
|
||||||
<dependentAssembly>
|
<dependentAssembly>
|
||||||
<assemblyIdentity name="CefSharp" publicKeyToken="40c4b6fc221f4138" culture="neutral" />
|
<assemblyIdentity name="CefSharp" publicKeyToken="40c4b6fc221f4138" culture="neutral" />
|
||||||
<bindingRedirect oldVersion="0.0.0.0-110.0.300.0" newVersion="110.0.300.0" />
|
<bindingRedirect oldVersion="0.0.0.0-118.6.80.0" newVersion="118.6.80.0" />
|
||||||
</dependentAssembly>
|
</dependentAssembly>
|
||||||
<dependentAssembly>
|
<dependentAssembly>
|
||||||
<assemblyIdentity name="CefSharp.Core" publicKeyToken="40c4b6fc221f4138" culture="neutral" />
|
<assemblyIdentity name="CefSharp.Core" publicKeyToken="40c4b6fc221f4138" culture="neutral" />
|
||||||
<bindingRedirect oldVersion="0.0.0.0-110.0.300.0" newVersion="110.0.300.0" />
|
<bindingRedirect oldVersion="0.0.0.0-118.6.80.0" newVersion="118.6.80.0" />
|
||||||
|
</dependentAssembly>
|
||||||
|
<dependentAssembly>
|
||||||
|
<assemblyIdentity name="NuGet.Frameworks" publicKeyToken="31bf3856ad364e35" culture="neutral" />
|
||||||
|
<bindingRedirect oldVersion="0.0.0.0-5.11.3.1" newVersion="5.11.3.1" />
|
||||||
|
</dependentAssembly>
|
||||||
|
<dependentAssembly>
|
||||||
|
<assemblyIdentity name="System.Collections.Immutable" publicKeyToken="b03f5f7f11d50a3a" culture="neutral" />
|
||||||
|
<bindingRedirect oldVersion="0.0.0.0-8.0.0.0" newVersion="8.0.0.0" />
|
||||||
|
</dependentAssembly>
|
||||||
|
<dependentAssembly>
|
||||||
|
<assemblyIdentity name="Microsoft.ApplicationInsights" publicKeyToken="31bf3856ad364e35" culture="neutral" />
|
||||||
|
<bindingRedirect oldVersion="0.0.0.0-2.22.0.997" newVersion="2.22.0.997" />
|
||||||
|
</dependentAssembly>
|
||||||
|
<dependentAssembly>
|
||||||
|
<assemblyIdentity name="System.Diagnostics.DiagnosticSource" publicKeyToken="cc7b13ffcd2ddd51" culture="neutral" />
|
||||||
|
<bindingRedirect oldVersion="0.0.0.0-8.0.0.0" newVersion="8.0.0.0" />
|
||||||
|
</dependentAssembly>
|
||||||
|
<dependentAssembly>
|
||||||
|
<assemblyIdentity name="System.Memory" publicKeyToken="cc7b13ffcd2ddd51" culture="neutral" />
|
||||||
|
<bindingRedirect oldVersion="0.0.0.0-4.0.1.2" newVersion="4.0.1.2" />
|
||||||
|
</dependentAssembly>
|
||||||
|
<dependentAssembly>
|
||||||
|
<assemblyIdentity name="System.Reflection.Metadata" publicKeyToken="b03f5f7f11d50a3a" culture="neutral" />
|
||||||
|
<bindingRedirect oldVersion="0.0.0.0-8.0.0.0" newVersion="8.0.0.0" />
|
||||||
</dependentAssembly>
|
</dependentAssembly>
|
||||||
</assemblyBinding>
|
</assemblyBinding>
|
||||||
</runtime>
|
</runtime>
|
||||||
</configuration>
|
<startup><supportedRuntime version="v4.0" sku=".NETFramework,Version=v4.8" /></startup></configuration>
|
||||||
|
|
|
@ -1,12 +1,26 @@
|
||||||
<?xml version="1.0" encoding="utf-8"?>
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
<packages>
|
<packages>
|
||||||
<package id="Castle.Core" version="5.1.1" targetFramework="net472" />
|
<package id="Castle.Core" version="5.1.1" targetFramework="net48" />
|
||||||
<package id="cef.redist.x64" version="111.2.2" targetFramework="net472" />
|
<package id="CefSharp.Common" version="121.3.130" targetFramework="net48" />
|
||||||
<package id="cef.redist.x86" version="111.2.2" targetFramework="net472" />
|
<package id="chromiumembeddedframework.runtime.win-x64" version="121.3.13" targetFramework="net48" />
|
||||||
<package id="CefSharp.Common" version="111.2.20" targetFramework="net472" />
|
<package id="chromiumembeddedframework.runtime.win-x86" version="121.3.13" targetFramework="net48" />
|
||||||
<package id="Moq" version="4.18.4" targetFramework="net472" />
|
<package id="Microsoft.ApplicationInsights" version="2.22.0" targetFramework="net48" />
|
||||||
<package id="MSTest.TestAdapter" version="3.0.2" targetFramework="net472" />
|
<package id="Microsoft.Testing.Extensions.Telemetry" version="1.0.2" targetFramework="net48" />
|
||||||
<package id="MSTest.TestFramework" version="3.0.2" targetFramework="net472" />
|
<package id="Microsoft.Testing.Extensions.TrxReport.Abstractions" version="1.0.2" targetFramework="net48" />
|
||||||
<package id="System.Runtime.CompilerServices.Unsafe" version="6.0.0" targetFramework="net472" />
|
<package id="Microsoft.Testing.Extensions.VSTestBridge" version="1.0.2" targetFramework="net48" />
|
||||||
<package id="System.Threading.Tasks.Extensions" version="4.5.4" targetFramework="net472" />
|
<package id="Microsoft.Testing.Platform" version="1.0.2" targetFramework="net48" />
|
||||||
|
<package id="Microsoft.Testing.Platform.MSBuild" version="1.0.2" targetFramework="net48" />
|
||||||
|
<package id="Microsoft.TestPlatform.ObjectModel" version="17.9.0" targetFramework="net48" />
|
||||||
|
<package id="Moq" version="4.20.70" targetFramework="net48" />
|
||||||
|
<package id="MSTest.TestAdapter" version="3.2.2" targetFramework="net48" />
|
||||||
|
<package id="MSTest.TestFramework" version="3.2.2" targetFramework="net48" />
|
||||||
|
<package id="NuGet.Frameworks" version="6.9.1" targetFramework="net48" />
|
||||||
|
<package id="System.Buffers" version="4.5.1" targetFramework="net48" />
|
||||||
|
<package id="System.Collections.Immutable" version="8.0.0" targetFramework="net48" />
|
||||||
|
<package id="System.Diagnostics.DiagnosticSource" version="8.0.0" targetFramework="net48" />
|
||||||
|
<package id="System.Memory" version="4.5.5" targetFramework="net48" />
|
||||||
|
<package id="System.Numerics.Vectors" version="4.5.0" targetFramework="net48" />
|
||||||
|
<package id="System.Reflection.Metadata" version="8.0.0" targetFramework="net48" />
|
||||||
|
<package id="System.Runtime.CompilerServices.Unsafe" version="6.0.0" targetFramework="net48" />
|
||||||
|
<package id="System.Threading.Tasks.Extensions" version="4.5.4" targetFramework="net48" />
|
||||||
</packages>
|
</packages>
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
@ -14,7 +14,6 @@ using System.Linq;
|
||||||
using System.Threading;
|
using System.Threading;
|
||||||
using CefSharp;
|
using CefSharp;
|
||||||
using CefSharp.WinForms;
|
using CefSharp.WinForms;
|
||||||
using SafeExamBrowser.Applications.Contracts;
|
|
||||||
using SafeExamBrowser.Applications.Contracts.Events;
|
using SafeExamBrowser.Applications.Contracts.Events;
|
||||||
using SafeExamBrowser.Browser.Contracts;
|
using SafeExamBrowser.Browser.Contracts;
|
||||||
using SafeExamBrowser.Browser.Contracts.Events;
|
using SafeExamBrowser.Browser.Contracts.Events;
|
||||||
|
@ -40,6 +39,7 @@ namespace SafeExamBrowser.Browser
|
||||||
private int windowIdCounter = default;
|
private int windowIdCounter = default;
|
||||||
|
|
||||||
private readonly AppConfig appConfig;
|
private readonly AppConfig appConfig;
|
||||||
|
private readonly Clipboard clipboard;
|
||||||
private readonly IFileSystemDialog fileSystemDialog;
|
private readonly IFileSystemDialog fileSystemDialog;
|
||||||
private readonly IHashAlgorithm hashAlgorithm;
|
private readonly IHashAlgorithm hashAlgorithm;
|
||||||
private readonly IKeyGenerator keyGenerator;
|
private readonly IKeyGenerator keyGenerator;
|
||||||
|
@ -59,9 +59,9 @@ namespace SafeExamBrowser.Browser
|
||||||
public string Tooltip { get; private set; }
|
public string Tooltip { get; private set; }
|
||||||
|
|
||||||
public event DownloadRequestedEventHandler ConfigurationDownloadRequested;
|
public event DownloadRequestedEventHandler ConfigurationDownloadRequested;
|
||||||
public event SessionIdentifierDetectedEventHandler SessionIdentifierDetected;
|
|
||||||
public event LoseFocusRequestedEventHandler LoseFocusRequested;
|
public event LoseFocusRequestedEventHandler LoseFocusRequested;
|
||||||
public event TerminationRequestedEventHandler TerminationRequested;
|
public event TerminationRequestedEventHandler TerminationRequested;
|
||||||
|
public event UserIdentifierDetectedEventHandler UserIdentifierDetected;
|
||||||
public event WindowsChangedEventHandler WindowsChanged;
|
public event WindowsChangedEventHandler WindowsChanged;
|
||||||
|
|
||||||
public BrowserApplication(
|
public BrowserApplication(
|
||||||
|
@ -78,6 +78,7 @@ namespace SafeExamBrowser.Browser
|
||||||
IUserInterfaceFactory uiFactory)
|
IUserInterfaceFactory uiFactory)
|
||||||
{
|
{
|
||||||
this.appConfig = appConfig;
|
this.appConfig = appConfig;
|
||||||
|
this.clipboard = new Clipboard(logger.CloneFor(nameof(Clipboard)), settings);
|
||||||
this.fileSystemDialog = fileSystemDialog;
|
this.fileSystemDialog = fileSystemDialog;
|
||||||
this.hashAlgorithm = hashAlgorithm;
|
this.hashAlgorithm = hashAlgorithm;
|
||||||
this.keyGenerator = keyGenerator;
|
this.keyGenerator = keyGenerator;
|
||||||
|
@ -99,9 +100,9 @@ namespace SafeExamBrowser.Browser
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
public IEnumerable<IApplicationWindow> GetWindows()
|
public IEnumerable<IBrowserWindow> GetWindows()
|
||||||
{
|
{
|
||||||
return new List<IApplicationWindow>(windows);
|
return new List<IBrowserWindow>(windows);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void Initialize()
|
public void Initialize()
|
||||||
|
@ -192,6 +193,7 @@ namespace SafeExamBrowser.Browser
|
||||||
var windowLogger = logger.CloneFor($"Browser Window #{id}");
|
var windowLogger = logger.CloneFor($"Browser Window #{id}");
|
||||||
var window = new BrowserWindow(
|
var window = new BrowserWindow(
|
||||||
appConfig,
|
appConfig,
|
||||||
|
clipboard,
|
||||||
fileSystemDialog,
|
fileSystemDialog,
|
||||||
hashAlgorithm,
|
hashAlgorithm,
|
||||||
id,
|
id,
|
||||||
|
@ -209,7 +211,7 @@ namespace SafeExamBrowser.Browser
|
||||||
window.ConfigurationDownloadRequested += (f, a) => ConfigurationDownloadRequested?.Invoke(f, a);
|
window.ConfigurationDownloadRequested += (f, a) => ConfigurationDownloadRequested?.Invoke(f, a);
|
||||||
window.PopupRequested += Window_PopupRequested;
|
window.PopupRequested += Window_PopupRequested;
|
||||||
window.ResetRequested += Window_ResetRequested;
|
window.ResetRequested += Window_ResetRequested;
|
||||||
window.SessionIdentifierDetected += (i) => SessionIdentifierDetected?.Invoke(i);
|
window.UserIdentifierDetected += (i) => UserIdentifierDetected?.Invoke(i);
|
||||||
window.TerminationRequested += () => TerminationRequested?.Invoke();
|
window.TerminationRequested += () => TerminationRequested?.Invoke();
|
||||||
window.LoseFocusRequested += (forward) => LoseFocusRequested?.Invoke(forward);
|
window.LoseFocusRequested += (forward) => LoseFocusRequested?.Invoke(forward);
|
||||||
|
|
||||||
|
@ -337,6 +339,11 @@ namespace SafeExamBrowser.Browser
|
||||||
cefSettings.PersistSessionCookies = !settings.DeleteCookiesOnStartup || !settings.DeleteCookiesOnShutdown;
|
cefSettings.PersistSessionCookies = !settings.DeleteCookiesOnStartup || !settings.DeleteCookiesOnShutdown;
|
||||||
cefSettings.UserAgent = InitializeUserAgent();
|
cefSettings.UserAgent = InitializeUserAgent();
|
||||||
|
|
||||||
|
if (!settings.AllowPageZoom)
|
||||||
|
{
|
||||||
|
cefSettings.CefCommandLineArgs.Add("disable-pinch");
|
||||||
|
}
|
||||||
|
|
||||||
if (!settings.AllowPdfReader)
|
if (!settings.AllowPdfReader)
|
||||||
{
|
{
|
||||||
cefSettings.CefCommandLineArgs.Add("disable-pdf-extension");
|
cefSettings.CefCommandLineArgs.Add("disable-pdf-extension");
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
@ -7,10 +7,12 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
using System;
|
using System;
|
||||||
|
using System.Linq;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
using CefSharp;
|
using CefSharp;
|
||||||
using SafeExamBrowser.Browser.Wrapper;
|
using SafeExamBrowser.Browser.Wrapper;
|
||||||
using SafeExamBrowser.Browser.Wrapper.Events;
|
using SafeExamBrowser.Browser.Wrapper.Events;
|
||||||
|
using SafeExamBrowser.Logging.Contracts;
|
||||||
using SafeExamBrowser.UserInterface.Contracts.Browser;
|
using SafeExamBrowser.UserInterface.Contracts.Browser;
|
||||||
using SafeExamBrowser.UserInterface.Contracts.Browser.Data;
|
using SafeExamBrowser.UserInterface.Contracts.Browser.Data;
|
||||||
using SafeExamBrowser.UserInterface.Contracts.Browser.Events;
|
using SafeExamBrowser.UserInterface.Contracts.Browser.Events;
|
||||||
|
@ -19,11 +21,13 @@ namespace SafeExamBrowser.Browser
|
||||||
{
|
{
|
||||||
internal class BrowserControl : IBrowserControl
|
internal class BrowserControl : IBrowserControl
|
||||||
{
|
{
|
||||||
|
private readonly Clipboard clipboard;
|
||||||
private readonly ICefSharpControl control;
|
private readonly ICefSharpControl control;
|
||||||
private readonly IDialogHandler dialogHandler;
|
private readonly IDialogHandler dialogHandler;
|
||||||
private readonly IDisplayHandler displayHandler;
|
private readonly IDisplayHandler displayHandler;
|
||||||
private readonly IDownloadHandler downloadHandler;
|
private readonly IDownloadHandler downloadHandler;
|
||||||
private readonly IKeyboardHandler keyboardHandler;
|
private readonly IKeyboardHandler keyboardHandler;
|
||||||
|
private readonly ILogger logger;
|
||||||
private readonly IRenderProcessMessageHandler renderProcessMessageHandler;
|
private readonly IRenderProcessMessageHandler renderProcessMessageHandler;
|
||||||
private readonly IRequestHandler requestHandler;
|
private readonly IRequestHandler requestHandler;
|
||||||
|
|
||||||
|
@ -38,19 +42,23 @@ namespace SafeExamBrowser.Browser
|
||||||
public event TitleChangedEventHandler TitleChanged;
|
public event TitleChangedEventHandler TitleChanged;
|
||||||
|
|
||||||
public BrowserControl(
|
public BrowserControl(
|
||||||
|
Clipboard clipboard,
|
||||||
ICefSharpControl control,
|
ICefSharpControl control,
|
||||||
IDialogHandler dialogHandler,
|
IDialogHandler dialogHandler,
|
||||||
IDisplayHandler displayHandler,
|
IDisplayHandler displayHandler,
|
||||||
IDownloadHandler downloadHandler,
|
IDownloadHandler downloadHandler,
|
||||||
IKeyboardHandler keyboardHandler,
|
IKeyboardHandler keyboardHandler,
|
||||||
|
ILogger logger,
|
||||||
IRenderProcessMessageHandler renderProcessMessageHandler,
|
IRenderProcessMessageHandler renderProcessMessageHandler,
|
||||||
IRequestHandler requestHandler)
|
IRequestHandler requestHandler)
|
||||||
{
|
{
|
||||||
this.control = control;
|
this.control = control;
|
||||||
|
this.clipboard = clipboard;
|
||||||
this.dialogHandler = dialogHandler;
|
this.dialogHandler = dialogHandler;
|
||||||
this.displayHandler = displayHandler;
|
this.displayHandler = displayHandler;
|
||||||
this.downloadHandler = downloadHandler;
|
this.downloadHandler = downloadHandler;
|
||||||
this.keyboardHandler = keyboardHandler;
|
this.keyboardHandler = keyboardHandler;
|
||||||
|
this.logger = logger;
|
||||||
this.renderProcessMessageHandler = renderProcessMessageHandler;
|
this.renderProcessMessageHandler = renderProcessMessageHandler;
|
||||||
this.requestHandler = requestHandler;
|
this.requestHandler = requestHandler;
|
||||||
}
|
}
|
||||||
|
@ -63,13 +71,15 @@ namespace SafeExamBrowser.Browser
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public void ExecuteJavascript(string javascript, Action<JavascriptResult> callback)
|
public void ExecuteJavaScript(string code, Action<JavaScriptResult> callback = default)
|
||||||
{
|
{
|
||||||
if ((control as IWebBrowser)?.CanExecuteJavascriptInMainFrame == true)
|
try
|
||||||
{
|
{
|
||||||
control.EvaluateScriptAsync(javascript).ContinueWith(t =>
|
if (control.BrowserCore != default && control.BrowserCore.MainFrame != default)
|
||||||
{
|
{
|
||||||
callback(new JavascriptResult
|
control.BrowserCore.EvaluateScriptAsync(code).ContinueWith(t =>
|
||||||
|
{
|
||||||
|
callback?.Invoke(new JavaScriptResult
|
||||||
{
|
{
|
||||||
Message = t.Result.Message,
|
Message = t.Result.Message,
|
||||||
Result = t.Result.Result,
|
Result = t.Result.Result,
|
||||||
|
@ -79,13 +89,23 @@ namespace SafeExamBrowser.Browser
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
Task.Run(() => callback(new JavascriptResult
|
Task.Run(() => callback?.Invoke(new JavaScriptResult
|
||||||
{
|
{
|
||||||
Message = "JavaScript can't be executed in main frame!",
|
Message = "JavaScript can't be executed in main frame!",
|
||||||
Success = false
|
Success = false
|
||||||
}));
|
}));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
catch (Exception e)
|
||||||
|
{
|
||||||
|
logger.Error($"Failed to execute JavaScript '{(code.Length > 50 ? code.Take(50) : code)}'!", e);
|
||||||
|
Task.Run(() => callback?.Invoke(new JavaScriptResult
|
||||||
|
{
|
||||||
|
Message = $"Failed to execute JavaScript '{(code.Length > 50 ? code.Take(50) : code)}'! Reason: {e.Message}",
|
||||||
|
Success = false
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public void Find(string term, bool isInitial, bool caseSensitive, bool forward = true)
|
public void Find(string term, bool isInitial, bool caseSensitive, bool forward = true)
|
||||||
{
|
{
|
||||||
|
@ -94,6 +114,8 @@ namespace SafeExamBrowser.Browser
|
||||||
|
|
||||||
public void Initialize()
|
public void Initialize()
|
||||||
{
|
{
|
||||||
|
clipboard.Changed += Clipboard_Changed;
|
||||||
|
|
||||||
control.AddressChanged += (o, e) => AddressChanged?.Invoke(e.Address);
|
control.AddressChanged += (o, e) => AddressChanged?.Invoke(e.Address);
|
||||||
control.AuthCredentialsRequired += (w, b, o, i, h, p, r, s, c, a) => a.Value = requestHandler.GetAuthCredentials(w, b, o, i, h, p, r, s, c);
|
control.AuthCredentialsRequired += (w, b, o, i, h, p, r, s, c, a) => a.Value = requestHandler.GetAuthCredentials(w, b, o, i, h, p, r, s, c);
|
||||||
control.BeforeBrowse += (w, b, f, r, u, i, a) => a.Value = requestHandler.OnBeforeBrowse(w, b, f, r, u, i);
|
control.BeforeBrowse += (w, b, f, r, u, i, a) => a.Value = requestHandler.OnBeforeBrowse(w, b, f, r, u, i);
|
||||||
|
@ -115,6 +137,11 @@ namespace SafeExamBrowser.Browser
|
||||||
control.ResourceRequestHandlerRequired += (IWebBrowser w, IBrowser b, IFrame f, IRequest r, bool n, bool d, string i, ref bool h, ResourceRequestEventArgs a) => a.Handler = requestHandler.GetResourceRequestHandler(w, b, f, r, n, d, i, ref h);
|
control.ResourceRequestHandlerRequired += (IWebBrowser w, IBrowser b, IFrame f, IRequest r, bool n, bool d, string i, ref bool h, ResourceRequestEventArgs a) => a.Handler = requestHandler.GetResourceRequestHandler(w, b, f, r, n, d, i, ref h);
|
||||||
control.TitleChanged += (o, e) => TitleChanged?.Invoke(e.Title);
|
control.TitleChanged += (o, e) => TitleChanged?.Invoke(e.Title);
|
||||||
control.UncaughtExceptionEvent += (w, b, f, e) => renderProcessMessageHandler.OnUncaughtException(w, b, f, e);
|
control.UncaughtExceptionEvent += (w, b, f, e) => renderProcessMessageHandler.OnUncaughtException(w, b, f, e);
|
||||||
|
|
||||||
|
if (control is IWebBrowser webBrowser)
|
||||||
|
{
|
||||||
|
webBrowser.JavascriptMessageReceived += WebBrowser_JavascriptMessageReceived;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public void NavigateBackwards()
|
public void NavigateBackwards()
|
||||||
|
@ -147,6 +174,11 @@ namespace SafeExamBrowser.Browser
|
||||||
control.BrowserCore.SetZoomLevel(level);
|
control.BrowserCore.SetZoomLevel(level);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void Clipboard_Changed(long id)
|
||||||
|
{
|
||||||
|
ExecuteJavaScript($"SafeExamBrowser.clipboard.update({id}, '{clipboard.Content}');");
|
||||||
|
}
|
||||||
|
|
||||||
private void Control_IsBrowserInitializedChanged(object sender, EventArgs e)
|
private void Control_IsBrowserInitializedChanged(object sender, EventArgs e)
|
||||||
{
|
{
|
||||||
if (control.IsBrowserInitialized)
|
if (control.IsBrowserInitialized)
|
||||||
|
@ -154,5 +186,10 @@ namespace SafeExamBrowser.Browser
|
||||||
control.BrowserCore.GetHost().SetFocus(true);
|
control.BrowserCore.GetHost().SetFocus(true);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void WebBrowser_JavascriptMessageReceived(object sender, JavascriptMessageReceivedEventArgs e)
|
||||||
|
{
|
||||||
|
clipboard.Process(e);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
@ -13,7 +13,6 @@ using System.Threading.Tasks;
|
||||||
using CefSharp;
|
using CefSharp;
|
||||||
using CefSharp.WinForms.Handler;
|
using CefSharp.WinForms.Handler;
|
||||||
using CefSharp.WinForms.Host;
|
using CefSharp.WinForms.Host;
|
||||||
using SafeExamBrowser.Applications.Contracts;
|
|
||||||
using SafeExamBrowser.Applications.Contracts.Events;
|
using SafeExamBrowser.Applications.Contracts.Events;
|
||||||
using SafeExamBrowser.Browser.Contracts.Events;
|
using SafeExamBrowser.Browser.Contracts.Events;
|
||||||
using SafeExamBrowser.Browser.Contracts.Filters;
|
using SafeExamBrowser.Browser.Contracts.Filters;
|
||||||
|
@ -43,16 +42,16 @@ using TitleChangedEventHandler = SafeExamBrowser.Applications.Contracts.Events.T
|
||||||
|
|
||||||
namespace SafeExamBrowser.Browser
|
namespace SafeExamBrowser.Browser
|
||||||
{
|
{
|
||||||
internal class BrowserWindow : IApplicationWindow
|
internal class BrowserWindow : Contracts.IBrowserWindow
|
||||||
{
|
{
|
||||||
private const string CLEAR_FIND_TERM = "thisisahacktoclearthesearchresultsasitappearsthatthereisnosuchfunctionalityincef";
|
private const string CLEAR_FIND_TERM = "thisisahacktoclearthesearchresultsasitappearsthatthereisnosuchfunctionalityincef";
|
||||||
private const double ZOOM_FACTOR = 0.2;
|
private const double ZOOM_FACTOR = 0.2;
|
||||||
|
|
||||||
private readonly AppConfig appConfig;
|
private readonly AppConfig appConfig;
|
||||||
|
private readonly Clipboard clipboard;
|
||||||
private readonly IFileSystemDialog fileSystemDialog;
|
private readonly IFileSystemDialog fileSystemDialog;
|
||||||
private readonly IHashAlgorithm hashAlgorithm;
|
private readonly IHashAlgorithm hashAlgorithm;
|
||||||
private readonly HttpClient httpClient;
|
private readonly HttpClient httpClient;
|
||||||
private readonly bool isMainWindow;
|
|
||||||
private readonly IKeyGenerator keyGenerator;
|
private readonly IKeyGenerator keyGenerator;
|
||||||
private readonly IModuleLogger logger;
|
private readonly IModuleLogger logger;
|
||||||
private readonly IMessageBox messageBox;
|
private readonly IMessageBox messageBox;
|
||||||
|
@ -69,7 +68,7 @@ namespace SafeExamBrowser.Browser
|
||||||
|
|
||||||
private WindowSettings WindowSettings
|
private WindowSettings WindowSettings
|
||||||
{
|
{
|
||||||
get { return isMainWindow ? settings.MainWindow : settings.AdditionalWindow; }
|
get { return IsMainWindow ? settings.MainWindow : settings.AdditionalWindow; }
|
||||||
}
|
}
|
||||||
|
|
||||||
internal IBrowserControl Control { get; private set; }
|
internal IBrowserControl Control { get; private set; }
|
||||||
|
@ -77,21 +76,24 @@ namespace SafeExamBrowser.Browser
|
||||||
|
|
||||||
public IntPtr Handle { get; private set; }
|
public IntPtr Handle { get; private set; }
|
||||||
public IconResource Icon { get; private set; }
|
public IconResource Icon { get; private set; }
|
||||||
|
public bool IsMainWindow { get; private set; }
|
||||||
public string Title { get; private set; }
|
public string Title { get; private set; }
|
||||||
|
public string Url { get; private set; }
|
||||||
|
|
||||||
internal event WindowClosedEventHandler Closed;
|
internal event WindowClosedEventHandler Closed;
|
||||||
internal event DownloadRequestedEventHandler ConfigurationDownloadRequested;
|
internal event DownloadRequestedEventHandler ConfigurationDownloadRequested;
|
||||||
|
internal event LoseFocusRequestedEventHandler LoseFocusRequested;
|
||||||
internal event PopupRequestedEventHandler PopupRequested;
|
internal event PopupRequestedEventHandler PopupRequested;
|
||||||
internal event ResetRequestedEventHandler ResetRequested;
|
internal event ResetRequestedEventHandler ResetRequested;
|
||||||
internal event SessionIdentifierDetectedEventHandler SessionIdentifierDetected;
|
|
||||||
internal event LoseFocusRequestedEventHandler LoseFocusRequested;
|
|
||||||
internal event TerminationRequestedEventHandler TerminationRequested;
|
internal event TerminationRequestedEventHandler TerminationRequested;
|
||||||
|
internal event UserIdentifierDetectedEventHandler UserIdentifierDetected;
|
||||||
|
|
||||||
public event IconChangedEventHandler IconChanged;
|
public event IconChangedEventHandler IconChanged;
|
||||||
public event TitleChangedEventHandler TitleChanged;
|
public event TitleChangedEventHandler TitleChanged;
|
||||||
|
|
||||||
public BrowserWindow(
|
public BrowserWindow(
|
||||||
AppConfig appConfig,
|
AppConfig appConfig,
|
||||||
|
Clipboard clipboard,
|
||||||
IFileSystemDialog fileSystemDialog,
|
IFileSystemDialog fileSystemDialog,
|
||||||
IHashAlgorithm hashAlgorithm,
|
IHashAlgorithm hashAlgorithm,
|
||||||
int id,
|
int id,
|
||||||
|
@ -106,11 +108,12 @@ namespace SafeExamBrowser.Browser
|
||||||
IUserInterfaceFactory uiFactory)
|
IUserInterfaceFactory uiFactory)
|
||||||
{
|
{
|
||||||
this.appConfig = appConfig;
|
this.appConfig = appConfig;
|
||||||
|
this.clipboard = clipboard;
|
||||||
this.fileSystemDialog = fileSystemDialog;
|
this.fileSystemDialog = fileSystemDialog;
|
||||||
this.hashAlgorithm = hashAlgorithm;
|
this.hashAlgorithm = hashAlgorithm;
|
||||||
this.httpClient = new HttpClient();
|
this.httpClient = new HttpClient();
|
||||||
this.Id = id;
|
this.Id = id;
|
||||||
this.isMainWindow = isMainWindow;
|
this.IsMainWindow = isMainWindow;
|
||||||
this.keyGenerator = keyGenerator;
|
this.keyGenerator = keyGenerator;
|
||||||
this.logger = logger;
|
this.logger = logger;
|
||||||
this.messageBox = messageBox;
|
this.messageBox = messageBox;
|
||||||
|
@ -149,12 +152,13 @@ namespace SafeExamBrowser.Browser
|
||||||
internal void InitializeControl()
|
internal void InitializeControl()
|
||||||
{
|
{
|
||||||
var cefSharpControl = default(ICefSharpControl);
|
var cefSharpControl = default(ICefSharpControl);
|
||||||
|
var controlLogger = logger.CloneFor($"{nameof(BrowserControl)} #{Id}");
|
||||||
var dialogHandler = new DialogHandler();
|
var dialogHandler = new DialogHandler();
|
||||||
var displayHandler = new DisplayHandler();
|
var displayHandler = new DisplayHandler();
|
||||||
var downloadLogger = logger.CloneFor($"{nameof(DownloadHandler)} #{Id}");
|
var downloadLogger = logger.CloneFor($"{nameof(DownloadHandler)} #{Id}");
|
||||||
var downloadHandler = new DownloadHandler(appConfig, downloadLogger, settings, WindowSettings);
|
var downloadHandler = new DownloadHandler(appConfig, downloadLogger, settings, WindowSettings);
|
||||||
var keyboardHandler = new KeyboardHandler();
|
var keyboardHandler = new KeyboardHandler();
|
||||||
var renderHandler = new RenderProcessMessageHandler(appConfig, keyGenerator, settings, text);
|
var renderHandler = new RenderProcessMessageHandler(appConfig, clipboard, keyGenerator, settings, text);
|
||||||
var requestFilter = new RequestFilter();
|
var requestFilter = new RequestFilter();
|
||||||
var requestLogger = logger.CloneFor($"{nameof(RequestHandler)} #{Id}");
|
var requestLogger = logger.CloneFor($"{nameof(RequestHandler)} #{Id}");
|
||||||
var resourceHandler = new ResourceHandler(appConfig, requestFilter, keyGenerator, logger, sessionMode, settings, WindowSettings, text);
|
var resourceHandler = new ResourceHandler(appConfig, requestFilter, keyGenerator, logger, sessionMode, settings, WindowSettings, text);
|
||||||
|
@ -162,7 +166,7 @@ namespace SafeExamBrowser.Browser
|
||||||
|
|
||||||
Icon = new BrowserIconResource();
|
Icon = new BrowserIconResource();
|
||||||
|
|
||||||
if (isMainWindow)
|
if (IsMainWindow)
|
||||||
{
|
{
|
||||||
cefSharpControl = new CefSharpBrowserControl(CreateLifeSpanHandlerForMainWindow(), startUrl);
|
cefSharpControl = new CefSharpBrowserControl(CreateLifeSpanHandlerForMainWindow(), startUrl);
|
||||||
}
|
}
|
||||||
|
@ -185,13 +189,13 @@ namespace SafeExamBrowser.Browser
|
||||||
keyboardHandler.ZoomInRequested += ZoomInRequested;
|
keyboardHandler.ZoomInRequested += ZoomInRequested;
|
||||||
keyboardHandler.ZoomOutRequested += ZoomOutRequested;
|
keyboardHandler.ZoomOutRequested += ZoomOutRequested;
|
||||||
keyboardHandler.ZoomResetRequested += ZoomResetRequested;
|
keyboardHandler.ZoomResetRequested += ZoomResetRequested;
|
||||||
resourceHandler.SessionIdentifierDetected += (id) => SessionIdentifierDetected?.Invoke(id);
|
|
||||||
requestHandler.QuitUrlVisited += RequestHandler_QuitUrlVisited;
|
requestHandler.QuitUrlVisited += RequestHandler_QuitUrlVisited;
|
||||||
requestHandler.RequestBlocked += RequestHandler_RequestBlocked;
|
requestHandler.RequestBlocked += RequestHandler_RequestBlocked;
|
||||||
|
resourceHandler.UserIdentifierDetected += (id) => UserIdentifierDetected?.Invoke(id);
|
||||||
|
|
||||||
InitializeRequestFilter(requestFilter);
|
InitializeRequestFilter(requestFilter);
|
||||||
|
|
||||||
Control = new BrowserControl(cefSharpControl, dialogHandler, displayHandler, downloadHandler, keyboardHandler, renderHandler, requestHandler);
|
Control = new BrowserControl(clipboard, cefSharpControl, dialogHandler, displayHandler, downloadHandler, keyboardHandler, controlLogger, renderHandler, requestHandler);
|
||||||
Control.AddressChanged += Control_AddressChanged;
|
Control.AddressChanged += Control_AddressChanged;
|
||||||
Control.LoadFailed += Control_LoadFailed;
|
Control.LoadFailed += Control_LoadFailed;
|
||||||
Control.LoadingStateChanged += Control_LoadingStateChanged;
|
Control.LoadingStateChanged += Control_LoadingStateChanged;
|
||||||
|
@ -203,7 +207,7 @@ namespace SafeExamBrowser.Browser
|
||||||
|
|
||||||
internal void InitializeWindow()
|
internal void InitializeWindow()
|
||||||
{
|
{
|
||||||
window = uiFactory.CreateBrowserWindow(Control, settings, isMainWindow, this.logger);
|
window = uiFactory.CreateBrowserWindow(Control, settings, IsMainWindow, this.logger);
|
||||||
window.AddressChanged += Window_AddressChanged;
|
window.AddressChanged += Window_AddressChanged;
|
||||||
window.BackwardNavigationRequested += Window_BackwardNavigationRequested;
|
window.BackwardNavigationRequested += Window_BackwardNavigationRequested;
|
||||||
window.Closed += Window_Closed;
|
window.Closed += Window_Closed;
|
||||||
|
@ -267,6 +271,8 @@ namespace SafeExamBrowser.Browser
|
||||||
private void Control_AddressChanged(string address)
|
private void Control_AddressChanged(string address)
|
||||||
{
|
{
|
||||||
logger.Info($"Navigated{(WindowSettings.UrlPolicy.CanLog() ? $" to '{address}'" : "")}.");
|
logger.Info($"Navigated{(WindowSettings.UrlPolicy.CanLog() ? $" to '{address}'" : "")}.");
|
||||||
|
|
||||||
|
Url = address;
|
||||||
window.UpdateAddress(address);
|
window.UpdateAddress(address);
|
||||||
|
|
||||||
if (WindowSettings.UrlPolicy == UrlPolicy.Always || WindowSettings.UrlPolicy == UrlPolicy.BeforeTitle)
|
if (WindowSettings.UrlPolicy == UrlPolicy.Always || WindowSettings.UrlPolicy == UrlPolicy.BeforeTitle)
|
||||||
|
@ -440,7 +446,7 @@ namespace SafeExamBrowser.Browser
|
||||||
|
|
||||||
private void HomeNavigationRequested()
|
private void HomeNavigationRequested()
|
||||||
{
|
{
|
||||||
if (isMainWindow && (settings.UseStartUrlAsHomeUrl || !string.IsNullOrWhiteSpace(settings.HomeUrl)))
|
if (IsMainWindow && (settings.UseStartUrlAsHomeUrl || !string.IsNullOrWhiteSpace(settings.HomeUrl)))
|
||||||
{
|
{
|
||||||
var navigate = false;
|
var navigate = false;
|
||||||
var url = settings.UseStartUrlAsHomeUrl ? settings.StartUrl : settings.HomeUrl;
|
var url = settings.UseStartUrlAsHomeUrl ? settings.StartUrl : settings.HomeUrl;
|
||||||
|
@ -497,7 +503,7 @@ namespace SafeExamBrowser.Browser
|
||||||
|
|
||||||
private void KeyboardHandler_TabPressed(bool shiftPressed)
|
private void KeyboardHandler_TabPressed(bool shiftPressed)
|
||||||
{
|
{
|
||||||
Control.ExecuteJavascript("document.activeElement.tagName", result =>
|
Control.ExecuteJavaScript("document.activeElement.tagName", result =>
|
||||||
{
|
{
|
||||||
if (result.Result is string tagName && tagName?.ToUpper() == "BODY")
|
if (result.Result is string tagName && tagName?.ToUpper() == "BODY")
|
||||||
{
|
{
|
||||||
|
|
72
SafeExamBrowser.Browser/Clipboard.cs
Normal file
72
SafeExamBrowser.Browser/Clipboard.cs
Normal file
|
@ -0,0 +1,72 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
|
*
|
||||||
|
* 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.Threading.Tasks;
|
||||||
|
using CefSharp;
|
||||||
|
using SafeExamBrowser.Browser.Events;
|
||||||
|
using SafeExamBrowser.Logging.Contracts;
|
||||||
|
using BrowserSettings = SafeExamBrowser.Settings.Browser.BrowserSettings;
|
||||||
|
|
||||||
|
namespace SafeExamBrowser.Browser
|
||||||
|
{
|
||||||
|
internal class Clipboard
|
||||||
|
{
|
||||||
|
private readonly ILogger logger;
|
||||||
|
private readonly BrowserSettings settings;
|
||||||
|
|
||||||
|
internal string Content { get; private set; }
|
||||||
|
|
||||||
|
internal event ClipboardChangedEventHandler Changed;
|
||||||
|
|
||||||
|
internal Clipboard(ILogger logger, BrowserSettings settings)
|
||||||
|
{
|
||||||
|
this.logger = logger;
|
||||||
|
this.settings = settings;
|
||||||
|
}
|
||||||
|
|
||||||
|
internal void Process(JavascriptMessageReceivedEventArgs message)
|
||||||
|
{
|
||||||
|
if (settings.UseIsolatedClipboard)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var data = message.ConvertMessageTo<Data>();
|
||||||
|
|
||||||
|
if (data != default && data.Type == "Clipboard" && TrySetContent(data.Content))
|
||||||
|
{
|
||||||
|
Task.Run(() => Changed?.Invoke(data.Id));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (Exception e)
|
||||||
|
{
|
||||||
|
logger.Error($"Failed to process browser message '{message}'!", e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private bool TrySetContent(object value)
|
||||||
|
{
|
||||||
|
var text = value as string;
|
||||||
|
|
||||||
|
if (text != default)
|
||||||
|
{
|
||||||
|
Content = text;
|
||||||
|
}
|
||||||
|
|
||||||
|
return text != default;
|
||||||
|
}
|
||||||
|
|
||||||
|
private class Data
|
||||||
|
{
|
||||||
|
public string Content { get; set; }
|
||||||
|
public long Id { get; set; }
|
||||||
|
public string Type { get; set; }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,4 +1,12 @@
|
||||||
SafeExamBrowser = {
|
/*
|
||||||
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
|
*
|
||||||
|
* 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/.
|
||||||
|
*/
|
||||||
|
|
||||||
|
SafeExamBrowser = {
|
||||||
version: 'SEB_Windows_%%_VERSION_%%',
|
version: 'SEB_Windows_%%_VERSION_%%',
|
||||||
security: {
|
security: {
|
||||||
browserExamKey: '%%_BEK_%%',
|
browserExamKey: '%%_BEK_%%',
|
||||||
|
|
195
SafeExamBrowser.Browser/Content/Clipboard.js
Normal file
195
SafeExamBrowser.Browser/Content/Clipboard.js
Normal file
|
@ -0,0 +1,195 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
|
*
|
||||||
|
* 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/.
|
||||||
|
*
|
||||||
|
* Original code taken and slightly adapted from https://github.com/eqsoft/seb2/blob/master/browser/app/modules/SebBrowser.jsm#L1215.
|
||||||
|
*/
|
||||||
|
|
||||||
|
SafeExamBrowser.clipboard = {
|
||||||
|
id: Math.round((Date.now() + Math.random()) * 1000),
|
||||||
|
ranges: [],
|
||||||
|
text: "",
|
||||||
|
|
||||||
|
clear: function () {
|
||||||
|
this.ranges = [];
|
||||||
|
this.text = "";
|
||||||
|
},
|
||||||
|
|
||||||
|
getContentEncoded: function () {
|
||||||
|
var bytes = new TextEncoder().encode(this.text);
|
||||||
|
var base64 = btoa(String.fromCodePoint(...bytes));
|
||||||
|
|
||||||
|
return base64;
|
||||||
|
},
|
||||||
|
|
||||||
|
update: function (id, base64) {
|
||||||
|
if (this.id != id) {
|
||||||
|
var bytes = Uint8Array.from(atob(base64), (m) => m.codePointAt(0));
|
||||||
|
var content = new TextDecoder().decode(bytes);
|
||||||
|
|
||||||
|
this.ranges = [];
|
||||||
|
this.text = content;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function copySelectedData(e) {
|
||||||
|
if (e.target.contentEditable && e.target.setRangeText) {
|
||||||
|
SafeExamBrowser.clipboard.text = e.target.value.substring(e.target.selectionStart, e.target.selectionEnd);
|
||||||
|
SafeExamBrowser.clipboard.ranges = [];
|
||||||
|
} else {
|
||||||
|
var selection = e.target.ownerDocument.defaultView.getSelection();
|
||||||
|
var text = "";
|
||||||
|
|
||||||
|
for (var i = 0; i < selection.rangeCount; i++) {
|
||||||
|
SafeExamBrowser.clipboard.ranges[i] = selection.getRangeAt(i).cloneContents();
|
||||||
|
text += SafeExamBrowser.clipboard.ranges[i].textContent;
|
||||||
|
}
|
||||||
|
|
||||||
|
SafeExamBrowser.clipboard.text = text;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function cutSelectedData(e) {
|
||||||
|
if (e.target.contentEditable && e.target.setRangeText) {
|
||||||
|
e.target.setRangeText("", e.target.selectionStart, e.target.selectionEnd, 'select');
|
||||||
|
} else {
|
||||||
|
var designMode = e.target.ownerDocument.designMode;
|
||||||
|
var contentEditables = e.target.ownerDocument.querySelectorAll('*[contenteditable]');
|
||||||
|
var selection = e.target.ownerDocument.defaultView.getSelection();
|
||||||
|
|
||||||
|
for (var i = 0; i < selection.rangeCount; i++) {
|
||||||
|
var range = selection.getRangeAt(i);
|
||||||
|
|
||||||
|
if (designMode === 'on') {
|
||||||
|
range.deleteContents();
|
||||||
|
} else {
|
||||||
|
if (contentEditables.length) {
|
||||||
|
contentEditables.forEach(node => {
|
||||||
|
if (node.contains(range.commonAncestorContainer)) {
|
||||||
|
range.deleteContents();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function pasteSelectedData(e) {
|
||||||
|
if (e.target.contentEditable && e.target.setRangeText) {
|
||||||
|
e.target.setRangeText("", e.target.selectionStart, e.target.selectionEnd, 'select');
|
||||||
|
e.target.setRangeText(SafeExamBrowser.clipboard.text, e.target.selectionStart, e.target.selectionStart + SafeExamBrowser.clipboard.text.length, 'end');
|
||||||
|
} else {
|
||||||
|
var w = e.target.ownerDocument.defaultView;
|
||||||
|
var designMode = e.target.ownerDocument.designMode;
|
||||||
|
var contentEditables = e.target.ownerDocument.querySelectorAll('*[contenteditable]');
|
||||||
|
var selection = w.getSelection();
|
||||||
|
|
||||||
|
for (var i = 0; i < selection.rangeCount; i++) {
|
||||||
|
var r = selection.getRangeAt(i);
|
||||||
|
|
||||||
|
if (designMode === 'on') {
|
||||||
|
r.deleteContents();
|
||||||
|
} else {
|
||||||
|
if (contentEditables.length) {
|
||||||
|
contentEditables.forEach(node => {
|
||||||
|
if (node.contains(r.commonAncestorContainer)) {
|
||||||
|
r.deleteContents();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (designMode === 'on') {
|
||||||
|
var range = w.getSelection().getRangeAt(0);
|
||||||
|
|
||||||
|
if (SafeExamBrowser.clipboard.ranges.length > 0) {
|
||||||
|
SafeExamBrowser.clipboard.ranges.map(r => {
|
||||||
|
range = w.getSelection().getRangeAt(0);
|
||||||
|
range.collapse();
|
||||||
|
const newNode = r.cloneNode(true);
|
||||||
|
range.insertNode(newNode);
|
||||||
|
range.collapse();
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
range.collapse();
|
||||||
|
range.insertNode(w.document.createTextNode(SafeExamBrowser.clipboard.text));
|
||||||
|
range.collapse();
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (contentEditables.length) {
|
||||||
|
contentEditables.forEach(node => {
|
||||||
|
var range = w.getSelection().getRangeAt(0);
|
||||||
|
|
||||||
|
if (node.contains(range.commonAncestorContainer)) {
|
||||||
|
if (SafeExamBrowser.clipboard.ranges.length > 0) {
|
||||||
|
SafeExamBrowser.clipboard.ranges.map(r => {
|
||||||
|
range = w.getSelection().getRangeAt(0);
|
||||||
|
range.collapse();
|
||||||
|
const newNode = r.cloneNode(true);
|
||||||
|
range.insertNode(newNode);
|
||||||
|
range.collapse();
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
range = w.getSelection().getRangeAt(0);
|
||||||
|
range.collapse();
|
||||||
|
range.insertNode(w.document.createTextNode(SafeExamBrowser.clipboard.text));
|
||||||
|
range.collapse();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function onCopy(e) {
|
||||||
|
SafeExamBrowser.clipboard.clear();
|
||||||
|
|
||||||
|
try {
|
||||||
|
copySelectedData(e);
|
||||||
|
|
||||||
|
CefSharp.PostMessage({ Type: "Clipboard", Id: SafeExamBrowser.clipboard.id, Content: SafeExamBrowser.clipboard.getContentEncoded() });
|
||||||
|
} finally {
|
||||||
|
e.preventDefault();
|
||||||
|
e.returnValue = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
function onCut(e) {
|
||||||
|
SafeExamBrowser.clipboard.clear();
|
||||||
|
|
||||||
|
try {
|
||||||
|
copySelectedData(e);
|
||||||
|
cutSelectedData(e);
|
||||||
|
|
||||||
|
CefSharp.PostMessage({ Type: "Clipboard", Id: SafeExamBrowser.clipboard.id, Content: SafeExamBrowser.clipboard.getContentEncoded() });
|
||||||
|
} finally {
|
||||||
|
e.preventDefault();
|
||||||
|
e.returnValue = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
function onPaste(e) {
|
||||||
|
try {
|
||||||
|
pasteSelectedData(e);
|
||||||
|
} finally {
|
||||||
|
e.preventDefault();
|
||||||
|
e.returnValue = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
window.document.addEventListener("copy", onCopy, true);
|
||||||
|
window.document.addEventListener("cut", onCut, true);
|
||||||
|
window.document.addEventListener("paste", onPaste, true);
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
@ -14,8 +14,11 @@ namespace SafeExamBrowser.Browser.Content
|
||||||
{
|
{
|
||||||
internal class ContentLoader
|
internal class ContentLoader
|
||||||
{
|
{
|
||||||
|
private readonly IText text;
|
||||||
|
|
||||||
private string api;
|
private string api;
|
||||||
private IText text;
|
private string clipboard;
|
||||||
|
private string pageZoom;
|
||||||
|
|
||||||
internal ContentLoader(IText text)
|
internal ContentLoader(IText text)
|
||||||
{
|
{
|
||||||
|
@ -24,7 +27,7 @@ namespace SafeExamBrowser.Browser.Content
|
||||||
|
|
||||||
internal string LoadApi(string browserExamKey, string configurationKey, string version)
|
internal string LoadApi(string browserExamKey, string configurationKey, string version)
|
||||||
{
|
{
|
||||||
if (api == default(string))
|
if (api == default)
|
||||||
{
|
{
|
||||||
var assembly = Assembly.GetAssembly(typeof(ContentLoader));
|
var assembly = Assembly.GetAssembly(typeof(ContentLoader));
|
||||||
var path = $"{typeof(ContentLoader).Namespace}.Api.js";
|
var path = $"{typeof(ContentLoader).Namespace}.Api.js";
|
||||||
|
@ -78,5 +81,39 @@ namespace SafeExamBrowser.Browser.Content
|
||||||
return html;
|
return html;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
internal string LoadClipboard()
|
||||||
|
{
|
||||||
|
if (clipboard == default)
|
||||||
|
{
|
||||||
|
var assembly = Assembly.GetAssembly(typeof(ContentLoader));
|
||||||
|
var path = $"{typeof(ContentLoader).Namespace}.Clipboard.js";
|
||||||
|
|
||||||
|
using (var stream = assembly.GetManifestResourceStream(path))
|
||||||
|
using (var reader = new StreamReader(stream))
|
||||||
|
{
|
||||||
|
clipboard = reader.ReadToEnd();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return clipboard;
|
||||||
|
}
|
||||||
|
|
||||||
|
internal string LoadPageZoom()
|
||||||
|
{
|
||||||
|
if (pageZoom == default)
|
||||||
|
{
|
||||||
|
var assembly = Assembly.GetAssembly(typeof(ContentLoader));
|
||||||
|
var path = $"{typeof(ContentLoader).Namespace}.PageZoom.js";
|
||||||
|
|
||||||
|
using (var stream = assembly.GetManifestResourceStream(path))
|
||||||
|
using (var reader = new StreamReader(stream))
|
||||||
|
{
|
||||||
|
pageZoom = reader.ReadToEnd();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return pageZoom;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
16
SafeExamBrowser.Browser/Content/PageZoom.js
Normal file
16
SafeExamBrowser.Browser/Content/PageZoom.js
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
|
*
|
||||||
|
* 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/.
|
||||||
|
*/
|
||||||
|
|
||||||
|
function disableMouseWheelZoom(e) {
|
||||||
|
if (e.ctrlKey) {
|
||||||
|
e.preventDefault();
|
||||||
|
e.stopPropagation();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
document.addEventListener('wheel', disableMouseWheelZoom, { passive: false });
|
|
@ -0,0 +1,12 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
|
*
|
||||||
|
* 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/.
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace SafeExamBrowser.Browser.Events
|
||||||
|
{
|
||||||
|
internal delegate void ClipboardChangedEventHandler(long id);
|
||||||
|
}
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
@ -138,6 +138,11 @@ namespace SafeExamBrowser.Browser.Handlers
|
||||||
filePath = Path.Combine(KnownFolders.Downloads.ExpandedPath, downloadItem.SuggestedFileName);
|
filePath = Path.Combine(KnownFolders.Downloads.ExpandedPath, downloadItem.SuggestedFileName);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (File.Exists(filePath))
|
||||||
|
{
|
||||||
|
filePath = AppendIndexSuffixTo(filePath);
|
||||||
|
}
|
||||||
|
|
||||||
if (showDialog)
|
if (showDialog)
|
||||||
{
|
{
|
||||||
logger.Debug($"Allowing user to select custom download location, with '{filePath}' as suggestion.");
|
logger.Debug($"Allowing user to select custom download location, with '{filePath}' as suggestion.");
|
||||||
|
@ -155,6 +160,26 @@ namespace SafeExamBrowser.Browser.Handlers
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private string AppendIndexSuffixTo(string filePath)
|
||||||
|
{
|
||||||
|
var directory = Path.GetDirectoryName(filePath);
|
||||||
|
var extension = Path.GetExtension(filePath);
|
||||||
|
var name = Path.GetFileNameWithoutExtension(filePath);
|
||||||
|
var path = default(string);
|
||||||
|
|
||||||
|
for (var suffix = 1; suffix < int.MaxValue; suffix++)
|
||||||
|
{
|
||||||
|
path = Path.Combine(directory, $"{name}({suffix}){extension}");
|
||||||
|
|
||||||
|
if (!File.Exists(path))
|
||||||
|
{
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return path;
|
||||||
|
}
|
||||||
|
|
||||||
private void RequestConfigurationFileDownload(DownloadItem downloadItem, IBeforeDownloadCallback callback)
|
private void RequestConfigurationFileDownload(DownloadItem downloadItem, IBeforeDownloadCallback callback)
|
||||||
{
|
{
|
||||||
var args = new DownloadEventArgs { Url = downloadItem.Url };
|
var args = new DownloadEventArgs { Url = downloadItem.Url };
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
@ -18,14 +18,16 @@ namespace SafeExamBrowser.Browser.Handlers
|
||||||
internal class RenderProcessMessageHandler : IRenderProcessMessageHandler
|
internal class RenderProcessMessageHandler : IRenderProcessMessageHandler
|
||||||
{
|
{
|
||||||
private readonly AppConfig appConfig;
|
private readonly AppConfig appConfig;
|
||||||
|
private readonly Clipboard clipboard;
|
||||||
private readonly ContentLoader contentLoader;
|
private readonly ContentLoader contentLoader;
|
||||||
private readonly IKeyGenerator keyGenerator;
|
private readonly IKeyGenerator keyGenerator;
|
||||||
private readonly BrowserSettings settings;
|
private readonly BrowserSettings settings;
|
||||||
private readonly IText text;
|
private readonly IText text;
|
||||||
|
|
||||||
internal RenderProcessMessageHandler(AppConfig appConfig, IKeyGenerator keyGenerator, BrowserSettings settings, IText text)
|
internal RenderProcessMessageHandler(AppConfig appConfig, Clipboard clipboard, IKeyGenerator keyGenerator, BrowserSettings settings, IText text)
|
||||||
{
|
{
|
||||||
this.appConfig = appConfig;
|
this.appConfig = appConfig;
|
||||||
|
this.clipboard = clipboard;
|
||||||
this.contentLoader = new ContentLoader(text);
|
this.contentLoader = new ContentLoader(text);
|
||||||
this.keyGenerator = keyGenerator;
|
this.keyGenerator = keyGenerator;
|
||||||
this.settings = settings;
|
this.settings = settings;
|
||||||
|
@ -37,12 +39,29 @@ namespace SafeExamBrowser.Browser.Handlers
|
||||||
var browserExamKey = keyGenerator.CalculateBrowserExamKeyHash(settings.ConfigurationKey, settings.BrowserExamKeySalt, frame.Url);
|
var browserExamKey = keyGenerator.CalculateBrowserExamKeyHash(settings.ConfigurationKey, settings.BrowserExamKeySalt, frame.Url);
|
||||||
var configurationKey = keyGenerator.CalculateConfigurationKeyHash(settings.ConfigurationKey, frame.Url);
|
var configurationKey = keyGenerator.CalculateConfigurationKeyHash(settings.ConfigurationKey, frame.Url);
|
||||||
var api = contentLoader.LoadApi(browserExamKey, configurationKey, appConfig.ProgramBuildVersion);
|
var api = contentLoader.LoadApi(browserExamKey, configurationKey, appConfig.ProgramBuildVersion);
|
||||||
|
var clipboardScript = contentLoader.LoadClipboard();
|
||||||
|
var pageZoomScript = contentLoader.LoadPageZoom();
|
||||||
|
|
||||||
frame.ExecuteJavaScriptAsync(api);
|
frame.ExecuteJavaScriptAsync(api);
|
||||||
|
|
||||||
|
if (!settings.AllowPageZoom)
|
||||||
|
{
|
||||||
|
frame.ExecuteJavaScriptAsync(pageZoomScript);
|
||||||
|
}
|
||||||
|
|
||||||
if (!settings.AllowPrint)
|
if (!settings.AllowPrint)
|
||||||
{
|
{
|
||||||
frame.ExecuteJavaScriptAsync($"window.print = function(){{ alert('{text.Get(TextKey.Browser_PrintNotAllowed)}') }}");
|
frame.ExecuteJavaScriptAsync($"window.print = function() {{ alert('{text.Get(TextKey.Browser_PrintNotAllowed)}') }}");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (settings.UseIsolatedClipboard)
|
||||||
|
{
|
||||||
|
frame.ExecuteJavaScriptAsync(clipboardScript);
|
||||||
|
|
||||||
|
if (clipboard.Content != default)
|
||||||
|
{
|
||||||
|
frame.ExecuteJavaScriptAsync($"SafeExamBrowser.clipboard.update('', '{clipboard.Content}');");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
@ -121,11 +121,26 @@ namespace SafeExamBrowser.Browser.Handlers
|
||||||
private bool IsConfigurationFile(IRequest request, out string downloadUrl)
|
private bool IsConfigurationFile(IRequest request, out string downloadUrl)
|
||||||
{
|
{
|
||||||
var isValidUri = Uri.TryCreate(request.Url, UriKind.RelativeOrAbsolute, out var uri);
|
var isValidUri = Uri.TryCreate(request.Url, UriKind.RelativeOrAbsolute, out var uri);
|
||||||
var isConfigurationFile = isValidUri && string.Equals(appConfig.ConfigurationFileExtension, Path.GetExtension(uri.AbsolutePath), StringComparison.OrdinalIgnoreCase);
|
var hasFileExtension = string.Equals(appConfig.ConfigurationFileExtension, Path.GetExtension(uri.AbsolutePath), StringComparison.OrdinalIgnoreCase);
|
||||||
|
var isDataUri = request.Url.Contains(appConfig.ConfigurationFileMimeType);
|
||||||
|
var isConfigurationFile = isValidUri && (hasFileExtension || isDataUri);
|
||||||
|
|
||||||
downloadUrl = request.Url;
|
downloadUrl = request.Url;
|
||||||
|
|
||||||
if (isConfigurationFile)
|
if (isConfigurationFile)
|
||||||
|
{
|
||||||
|
if (isDataUri)
|
||||||
|
{
|
||||||
|
if (uri.Scheme == appConfig.SebUriScheme)
|
||||||
|
{
|
||||||
|
downloadUrl = request.Url.Replace($"{appConfig.SebUriScheme}{Uri.SchemeDelimiter}", "data:");
|
||||||
|
}
|
||||||
|
else if (uri.Scheme == appConfig.SebUriSchemeSecure)
|
||||||
|
{
|
||||||
|
downloadUrl = request.Url.Replace($"{appConfig.SebUriSchemeSecure}{Uri.SchemeDelimiter}", "data:");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
{
|
{
|
||||||
if (uri.Scheme == appConfig.SebUriScheme)
|
if (uri.Scheme == appConfig.SebUriScheme)
|
||||||
{
|
{
|
||||||
|
@ -135,6 +150,7 @@ namespace SafeExamBrowser.Browser.Handlers
|
||||||
{
|
{
|
||||||
downloadUrl = new UriBuilder(uri) { Scheme = Uri.UriSchemeHttps }.Uri.AbsoluteUri;
|
downloadUrl = new UriBuilder(uri) { Scheme = Uri.UriSchemeHttps }.Uri.AbsoluteUri;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
logger.Debug($"Detected configuration file {(windowSettings.UrlPolicy.CanLog() ? $"'{uri}'" : "")}.");
|
logger.Debug($"Detected configuration file {(windowSettings.UrlPolicy.CanLog() ? $"'{uri}'" : "")}.");
|
||||||
}
|
}
|
||||||
|
@ -150,7 +166,7 @@ namespace SafeExamBrowser.Browser.Handlers
|
||||||
{
|
{
|
||||||
if (quitUrlPattern == default)
|
if (quitUrlPattern == default)
|
||||||
{
|
{
|
||||||
quitUrlPattern = Regex.Escape(settings.QuitUrl.TrimEnd('/')) + @"\/?";
|
quitUrlPattern = $"^{Regex.Escape(settings.QuitUrl.TrimEnd('/'))}/?$";
|
||||||
}
|
}
|
||||||
|
|
||||||
isQuitUrl = Regex.IsMatch(request.Url, quitUrlPattern, RegexOptions.IgnoreCase);
|
isQuitUrl = Regex.IsMatch(request.Url, quitUrlPattern, RegexOptions.IgnoreCase);
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
@ -44,9 +44,9 @@ namespace SafeExamBrowser.Browser.Handlers
|
||||||
|
|
||||||
private IResourceHandler contentHandler;
|
private IResourceHandler contentHandler;
|
||||||
private IResourceHandler pageHandler;
|
private IResourceHandler pageHandler;
|
||||||
private string sessionIdentifier;
|
private string userIdentifier;
|
||||||
|
|
||||||
internal event SessionIdentifierDetectedEventHandler SessionIdentifierDetected;
|
internal event UserIdentifierDetectedEventHandler UserIdentifierDetected;
|
||||||
|
|
||||||
internal ResourceHandler(
|
internal ResourceHandler(
|
||||||
AppConfig appConfig,
|
AppConfig appConfig,
|
||||||
|
@ -100,7 +100,7 @@ namespace SafeExamBrowser.Browser.Handlers
|
||||||
{
|
{
|
||||||
if (sessionMode == SessionMode.Server)
|
if (sessionMode == SessionMode.Server)
|
||||||
{
|
{
|
||||||
SearchSessionIdentifiers(request, response);
|
SearchUserIdentifier(request, response);
|
||||||
}
|
}
|
||||||
|
|
||||||
base.OnResourceRedirect(chromiumWebBrowser, browser, frame, request, response, ref newUrl);
|
base.OnResourceRedirect(chromiumWebBrowser, browser, frame, request, response, ref newUrl);
|
||||||
|
@ -117,7 +117,7 @@ namespace SafeExamBrowser.Browser.Handlers
|
||||||
|
|
||||||
if (sessionMode == SessionMode.Server)
|
if (sessionMode == SessionMode.Server)
|
||||||
{
|
{
|
||||||
SearchSessionIdentifiers(request, response);
|
SearchUserIdentifier(request, response);
|
||||||
}
|
}
|
||||||
|
|
||||||
return base.OnResourceResponse(webBrowser, browser, frame, request, response);
|
return base.OnResourceResponse(webBrowser, browser, frame, request, response);
|
||||||
|
@ -128,7 +128,7 @@ namespace SafeExamBrowser.Browser.Handlers
|
||||||
Uri.TryCreate(webBrowser.Address, UriKind.Absolute, out var pageUrl);
|
Uri.TryCreate(webBrowser.Address, UriKind.Absolute, out var pageUrl);
|
||||||
Uri.TryCreate(request.Url, UriKind.Absolute, out var requestUrl);
|
Uri.TryCreate(request.Url, UriKind.Absolute, out var requestUrl);
|
||||||
|
|
||||||
if (pageUrl?.Host?.Equals(requestUrl?.Host) == true)
|
if (request.ResourceType == ResourceType.MainFrame || pageUrl?.Host?.Equals(requestUrl?.Host) == true)
|
||||||
{
|
{
|
||||||
var headers = new NameValueCollection(request.Headers);
|
var headers = new NameValueCollection(request.Headers);
|
||||||
|
|
||||||
|
@ -233,41 +233,46 @@ namespace SafeExamBrowser.Browser.Handlers
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void SearchSessionIdentifiers(IRequest request, IResponse response)
|
private void SearchUserIdentifier(IRequest request, IResponse response)
|
||||||
{
|
{
|
||||||
var success = TrySearchGenericSessionIdentifier(response);
|
var success = TrySearchGenericUserIdentifier(response);
|
||||||
|
|
||||||
if (!success)
|
if (!success)
|
||||||
{
|
{
|
||||||
SearchEdxIdentifier(response);
|
success = TrySearchEdxUserIdentifier(response);
|
||||||
SearchMoodleIdentifier(request, response);
|
}
|
||||||
|
|
||||||
|
if (!success)
|
||||||
|
{
|
||||||
|
TrySearchMoodleUserIdentifier(request, response);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private bool TrySearchGenericSessionIdentifier(IResponse response)
|
private bool TrySearchGenericUserIdentifier(IResponse response)
|
||||||
{
|
{
|
||||||
var ids = response.Headers.GetValues("X-LMS-USER-ID");
|
var ids = response.Headers.GetValues("X-LMS-USER-ID");
|
||||||
|
var success = false;
|
||||||
|
|
||||||
if (ids != default(string[]))
|
if (ids != default(string[]))
|
||||||
{
|
{
|
||||||
var userId = ids.FirstOrDefault();
|
var userId = ids.FirstOrDefault();
|
||||||
|
|
||||||
if (userId != default && sessionIdentifier != userId)
|
if (userId != default && userIdentifier != userId)
|
||||||
{
|
{
|
||||||
sessionIdentifier = userId;
|
userIdentifier = userId;
|
||||||
Task.Run(() => SessionIdentifierDetected?.Invoke(sessionIdentifier));
|
Task.Run(() => UserIdentifierDetected?.Invoke(userIdentifier));
|
||||||
logger.Info("Generic LMS session detected.");
|
logger.Info("Generic LMS user identifier detected.");
|
||||||
|
success = true;
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return false;
|
return success;
|
||||||
}
|
}
|
||||||
|
|
||||||
private void SearchEdxIdentifier(IResponse response)
|
private bool TrySearchEdxUserIdentifier(IResponse response)
|
||||||
{
|
{
|
||||||
var cookies = response.Headers.GetValues("Set-Cookie");
|
var cookies = response.Headers.GetValues("Set-Cookie");
|
||||||
|
var success = false;
|
||||||
|
|
||||||
if (cookies != default(string[]))
|
if (cookies != default(string[]))
|
||||||
{
|
{
|
||||||
|
@ -284,32 +289,42 @@ namespace SafeExamBrowser.Browser.Handlers
|
||||||
var json = JsonConvert.DeserializeObject(sanitized) as JObject;
|
var json = JsonConvert.DeserializeObject(sanitized) as JObject;
|
||||||
var userName = json["username"].Value<string>();
|
var userName = json["username"].Value<string>();
|
||||||
|
|
||||||
if (sessionIdentifier != userName)
|
if (userIdentifier != userName)
|
||||||
{
|
{
|
||||||
sessionIdentifier = userName;
|
userIdentifier = userName;
|
||||||
Task.Run(() => SessionIdentifierDetected?.Invoke(sessionIdentifier));
|
Task.Run(() => UserIdentifierDetected?.Invoke(userIdentifier));
|
||||||
logger.Info("EdX session detected.");
|
logger.Info("EdX user identifier detected.");
|
||||||
|
success = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
catch (Exception e)
|
catch (Exception e)
|
||||||
{
|
{
|
||||||
logger.Error("Failed to parse edX session identifier!", e);
|
logger.Error("Failed to parse edX user identifier!", e);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void SearchMoodleIdentifier(IRequest request, IResponse response)
|
return success;
|
||||||
|
}
|
||||||
|
|
||||||
|
private bool TrySearchMoodleUserIdentifier(IRequest request, IResponse response)
|
||||||
{
|
{
|
||||||
var success = TrySearchByLocation(response);
|
var success = TrySearchMoodleUserIdentifierByLocation(response);
|
||||||
|
|
||||||
if (!success)
|
if (!success)
|
||||||
{
|
{
|
||||||
TrySearchBySession(request, response);
|
success = TrySearchMoodleUserIdentifierByRequest(MoodleRequestType.Plugin, request, response);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private bool TrySearchByLocation(IResponse response)
|
if (!success)
|
||||||
|
{
|
||||||
|
success = TrySearchMoodleUserIdentifierByRequest(MoodleRequestType.Theme, request, response);
|
||||||
|
}
|
||||||
|
|
||||||
|
return success;
|
||||||
|
}
|
||||||
|
|
||||||
|
private bool TrySearchMoodleUserIdentifierByLocation(IResponse response)
|
||||||
{
|
{
|
||||||
var locations = response.Headers.GetValues("Location");
|
var locations = response.Headers.GetValues("Location");
|
||||||
|
|
||||||
|
@ -323,11 +338,11 @@ namespace SafeExamBrowser.Browser.Handlers
|
||||||
{
|
{
|
||||||
var userId = location.Substring(location.IndexOf("=") + 1);
|
var userId = location.Substring(location.IndexOf("=") + 1);
|
||||||
|
|
||||||
if (sessionIdentifier != userId)
|
if (userIdentifier != userId)
|
||||||
{
|
{
|
||||||
sessionIdentifier = userId;
|
userIdentifier = userId;
|
||||||
Task.Run(() => SessionIdentifierDetected?.Invoke(sessionIdentifier));
|
Task.Run(() => UserIdentifierDetected?.Invoke(userIdentifier));
|
||||||
logger.Info("Moodle session detected.");
|
logger.Info("Moodle user identifier detected by location.");
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
|
@ -335,16 +350,17 @@ namespace SafeExamBrowser.Browser.Handlers
|
||||||
}
|
}
|
||||||
catch (Exception e)
|
catch (Exception e)
|
||||||
{
|
{
|
||||||
logger.Error("Failed to parse Moodle session identifier!", e);
|
logger.Error("Failed to parse Moodle user identifier by location!", e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
private void TrySearchBySession(IRequest request, IResponse response)
|
private bool TrySearchMoodleUserIdentifierByRequest(MoodleRequestType type, IRequest request, IResponse response)
|
||||||
{
|
{
|
||||||
var cookies = response.Headers.GetValues("Set-Cookie");
|
var cookies = response.Headers.GetValues("Set-Cookie");
|
||||||
|
var success = false;
|
||||||
|
|
||||||
if (cookies != default(string[]))
|
if (cookies != default(string[]))
|
||||||
{
|
{
|
||||||
|
@ -352,51 +368,84 @@ namespace SafeExamBrowser.Browser.Handlers
|
||||||
|
|
||||||
if (session != default)
|
if (session != default)
|
||||||
{
|
{
|
||||||
var requestUrl = request.Url;
|
var userId = ExecuteMoodleUserIdentifierRequest(request.Url, session, type);
|
||||||
|
|
||||||
|
if (int.TryParse(userId, out var id) && id > 0 && userIdentifier != userId)
|
||||||
|
{
|
||||||
|
userIdentifier = userId;
|
||||||
|
Task.Run(() => UserIdentifierDetected?.Invoke(userIdentifier));
|
||||||
|
logger.Info($"Moodle user identifier detected by request ({type}).");
|
||||||
|
success = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return success;
|
||||||
|
}
|
||||||
|
|
||||||
|
private string ExecuteMoodleUserIdentifierRequest(string requestUrl, string session, MoodleRequestType type)
|
||||||
|
{
|
||||||
|
var userId = default(string);
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
Task.Run(async () =>
|
Task.Run(async () =>
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
|
var endpointUrl = default(string);
|
||||||
var start = session.IndexOf("=") + 1;
|
var start = session.IndexOf("=") + 1;
|
||||||
var end = session.IndexOf(";");
|
var end = session.IndexOf(";");
|
||||||
|
var name = session.Substring(0, start - 1);
|
||||||
var value = session.Substring(start, end - start);
|
var value = session.Substring(start, end - start);
|
||||||
var uri = new Uri(requestUrl);
|
var uri = new Uri(requestUrl);
|
||||||
var message = new HttpRequestMessage(HttpMethod.Get, $"{uri.Scheme}{Uri.SchemeDelimiter}{uri.Host}/theme/boost_ethz/sebuser.php");
|
|
||||||
|
if (type == MoodleRequestType.Plugin)
|
||||||
|
{
|
||||||
|
endpointUrl = $"{uri.Scheme}{Uri.SchemeDelimiter}{uri.Host}/mod/quiz/accessrule/sebserver/classes/external/user.php";
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
endpointUrl = $"{uri.Scheme}{Uri.SchemeDelimiter}{uri.Host}/theme/boost_ethz/sebuser.php";
|
||||||
|
}
|
||||||
|
|
||||||
|
var message = new HttpRequestMessage(HttpMethod.Get, endpointUrl);
|
||||||
|
|
||||||
using (var handler = new HttpClientHandler { UseCookies = false })
|
using (var handler = new HttpClientHandler { UseCookies = false })
|
||||||
using (var client = new HttpClient(handler))
|
using (var client = new HttpClient(handler))
|
||||||
{
|
{
|
||||||
message.Headers.Add("Cookie", $"MoodleSession={value}");
|
message.Headers.Add("Cookie", $"{name}={value}");
|
||||||
|
|
||||||
var result = await client.SendAsync(message);
|
var result = await client.SendAsync(message);
|
||||||
|
|
||||||
if (result.IsSuccessStatusCode)
|
if (result.IsSuccessStatusCode)
|
||||||
{
|
{
|
||||||
var userId = await result.Content.ReadAsStringAsync();
|
userId = await result.Content.ReadAsStringAsync();
|
||||||
|
|
||||||
if (int.TryParse(userId, out var id) && id > 0 && sessionIdentifier != userId)
|
|
||||||
{
|
|
||||||
#pragma warning disable CS4014
|
|
||||||
sessionIdentifier = userId;
|
|
||||||
Task.Run(() => SessionIdentifierDetected?.Invoke(sessionIdentifier));
|
|
||||||
logger.Info("Moodle session detected.");
|
|
||||||
#pragma warning restore CS4014
|
|
||||||
}
|
}
|
||||||
}
|
else if (result.StatusCode != HttpStatusCode.NotFound)
|
||||||
else
|
|
||||||
{
|
{
|
||||||
logger.Error($"Failed to retrieve Moodle session identifier! Response: {result.StatusCode} {result.ReasonPhrase}");
|
logger.Error($"Failed to retrieve Moodle user identifier by request ({type})! Response: {(int) result.StatusCode} {result.ReasonPhrase}");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
catch (Exception e)
|
catch (Exception e)
|
||||||
{
|
{
|
||||||
logger.Error("Failed to parse Moodle session identifier!", e);
|
logger.Error($"Failed to parse Moodle user identifier by request ({type})!", e);
|
||||||
}
|
}
|
||||||
});
|
}).GetAwaiter().GetResult();
|
||||||
}
|
}
|
||||||
|
catch (Exception e)
|
||||||
|
{
|
||||||
|
logger.Error($"Failed to execute Moodle user identifier request ({type})!", e);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return userId;
|
||||||
|
}
|
||||||
|
|
||||||
|
private enum MoodleRequestType
|
||||||
|
{
|
||||||
|
Plugin,
|
||||||
|
Theme
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -9,7 +9,7 @@ using System.Runtime.InteropServices;
|
||||||
[assembly: AssemblyDescription("Safe Exam Browser")]
|
[assembly: AssemblyDescription("Safe Exam Browser")]
|
||||||
[assembly: AssemblyCompany("ETH Zürich")]
|
[assembly: AssemblyCompany("ETH Zürich")]
|
||||||
[assembly: AssemblyProduct("SafeExamBrowser.Browser")]
|
[assembly: AssemblyProduct("SafeExamBrowser.Browser")]
|
||||||
[assembly: AssemblyCopyright("Copyright © 2023 ETH Zürich, Educational Development and Technology (LET)")]
|
[assembly: AssemblyCopyright("Copyright © 2024 ETH Zürich, IT Services")]
|
||||||
|
|
||||||
// Setting ComVisible to false makes the types in this assembly not visible
|
// Setting ComVisible to false makes the types in this assembly not visible
|
||||||
// to COM components. If you need to access a type in this assembly from
|
// to COM components. If you need to access a type in this assembly from
|
||||||
|
|
|
@ -1,8 +1,8 @@
|
||||||
<?xml version="1.0" encoding="utf-8"?>
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
<Project ToolsVersion="15.0" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
|
<Project ToolsVersion="15.0" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
|
||||||
<Import Project="..\packages\CefSharp.Common.111.2.20\build\CefSharp.Common.props" Condition="Exists('..\packages\CefSharp.Common.111.2.20\build\CefSharp.Common.props')" />
|
<Import Project="..\packages\CefSharp.Common.121.3.130\build\CefSharp.Common.props" Condition="Exists('..\packages\CefSharp.Common.121.3.130\build\CefSharp.Common.props')" />
|
||||||
<Import Project="..\packages\cef.redist.x86.111.2.2\build\cef.redist.x86.props" Condition="Exists('..\packages\cef.redist.x86.111.2.2\build\cef.redist.x86.props')" />
|
<Import Project="..\packages\chromiumembeddedframework.runtime.win-x86.121.3.13\build\chromiumembeddedframework.runtime.win-x86.props" Condition="Exists('..\packages\chromiumembeddedframework.runtime.win-x86.121.3.13\build\chromiumembeddedframework.runtime.win-x86.props')" />
|
||||||
<Import Project="..\packages\cef.redist.x64.111.2.2\build\cef.redist.x64.props" Condition="Exists('..\packages\cef.redist.x64.111.2.2\build\cef.redist.x64.props')" />
|
<Import Project="..\packages\chromiumembeddedframework.runtime.win-x64.121.3.13\build\chromiumembeddedframework.runtime.win-x64.props" Condition="Exists('..\packages\chromiumembeddedframework.runtime.win-x64.121.3.13\build\chromiumembeddedframework.runtime.win-x64.props')" />
|
||||||
<Import Project="$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props" Condition="Exists('$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props')" />
|
<Import Project="$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props" Condition="Exists('$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props')" />
|
||||||
<PropertyGroup>
|
<PropertyGroup>
|
||||||
<Configuration Condition=" '$(Configuration)' == '' ">Debug</Configuration>
|
<Configuration Condition=" '$(Configuration)' == '' ">Debug</Configuration>
|
||||||
|
@ -12,7 +12,7 @@
|
||||||
<AppDesignerFolder>Properties</AppDesignerFolder>
|
<AppDesignerFolder>Properties</AppDesignerFolder>
|
||||||
<RootNamespace>SafeExamBrowser.Browser</RootNamespace>
|
<RootNamespace>SafeExamBrowser.Browser</RootNamespace>
|
||||||
<AssemblyName>SafeExamBrowser.Browser</AssemblyName>
|
<AssemblyName>SafeExamBrowser.Browser</AssemblyName>
|
||||||
<TargetFrameworkVersion>v4.7.2</TargetFrameworkVersion>
|
<TargetFrameworkVersion>v4.8</TargetFrameworkVersion>
|
||||||
<FileAlignment>512</FileAlignment>
|
<FileAlignment>512</FileAlignment>
|
||||||
<NuGetPackageImportStamp>
|
<NuGetPackageImportStamp>
|
||||||
</NuGetPackageImportStamp>
|
</NuGetPackageImportStamp>
|
||||||
|
@ -53,14 +53,14 @@
|
||||||
<ErrorReport>prompt</ErrorReport>
|
<ErrorReport>prompt</ErrorReport>
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<Reference Include="CefSharp, Version=111.2.20.0, Culture=neutral, PublicKeyToken=40c4b6fc221f4138, processorArchitecture=MSIL">
|
<Reference Include="CefSharp, Version=121.3.130.0, Culture=neutral, PublicKeyToken=40c4b6fc221f4138, processorArchitecture=MSIL">
|
||||||
<HintPath>..\packages\CefSharp.Common.111.2.20\lib\net452\CefSharp.dll</HintPath>
|
<HintPath>..\packages\CefSharp.Common.121.3.130\lib\net462\CefSharp.dll</HintPath>
|
||||||
</Reference>
|
</Reference>
|
||||||
<Reference Include="CefSharp.Core, Version=111.2.20.0, Culture=neutral, PublicKeyToken=40c4b6fc221f4138, processorArchitecture=MSIL">
|
<Reference Include="CefSharp.Core, Version=121.3.130.0, Culture=neutral, PublicKeyToken=40c4b6fc221f4138, processorArchitecture=MSIL">
|
||||||
<HintPath>..\packages\CefSharp.Common.111.2.20\lib\net452\CefSharp.Core.dll</HintPath>
|
<HintPath>..\packages\CefSharp.Common.121.3.130\lib\net462\CefSharp.Core.dll</HintPath>
|
||||||
</Reference>
|
</Reference>
|
||||||
<Reference Include="CefSharp.WinForms, Version=111.2.20.0, Culture=neutral, PublicKeyToken=40c4b6fc221f4138, processorArchitecture=MSIL">
|
<Reference Include="CefSharp.WinForms, Version=121.3.130.0, Culture=neutral, PublicKeyToken=40c4b6fc221f4138, processorArchitecture=MSIL">
|
||||||
<HintPath>..\packages\CefSharp.WinForms.111.2.20\lib\net462\CefSharp.WinForms.dll</HintPath>
|
<HintPath>..\packages\CefSharp.WinForms.121.3.130\lib\net462\CefSharp.WinForms.dll</HintPath>
|
||||||
</Reference>
|
</Reference>
|
||||||
<Reference Include="Newtonsoft.Json, Version=13.0.0.0, Culture=neutral, PublicKeyToken=30ad4fe6b2a6aeed, processorArchitecture=MSIL">
|
<Reference Include="Newtonsoft.Json, Version=13.0.0.0, Culture=neutral, PublicKeyToken=30ad4fe6b2a6aeed, processorArchitecture=MSIL">
|
||||||
<HintPath>..\packages\Newtonsoft.Json.13.0.3\lib\net45\Newtonsoft.Json.dll</HintPath>
|
<HintPath>..\packages\Newtonsoft.Json.13.0.3\lib\net45\Newtonsoft.Json.dll</HintPath>
|
||||||
|
@ -80,6 +80,8 @@
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<Compile Include="BrowserApplication.cs" />
|
<Compile Include="BrowserApplication.cs" />
|
||||||
<Compile Include="BrowserWindow.cs" />
|
<Compile Include="BrowserWindow.cs" />
|
||||||
|
<Compile Include="Clipboard.cs" />
|
||||||
|
<Compile Include="Events\ClipboardChangedEventHandler.cs" />
|
||||||
<Compile Include="Events\DialogRequestedEventArgs.cs" />
|
<Compile Include="Events\DialogRequestedEventArgs.cs" />
|
||||||
<Compile Include="Events\DialogRequestedEventHandler.cs" />
|
<Compile Include="Events\DialogRequestedEventHandler.cs" />
|
||||||
<Compile Include="Events\DownloadAbortedEventHandler.cs" />
|
<Compile Include="Events\DownloadAbortedEventHandler.cs" />
|
||||||
|
@ -189,6 +191,12 @@
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<EmbeddedResource Include="Content\Api.js" />
|
<EmbeddedResource Include="Content\Api.js" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
<ItemGroup>
|
||||||
|
<EmbeddedResource Include="Content\Clipboard.js" />
|
||||||
|
</ItemGroup>
|
||||||
|
<ItemGroup>
|
||||||
|
<EmbeddedResource Include="Content\PageZoom.js" />
|
||||||
|
</ItemGroup>
|
||||||
<Import Project="$(MSBuildToolsPath)\Microsoft.CSharp.targets" />
|
<Import Project="$(MSBuildToolsPath)\Microsoft.CSharp.targets" />
|
||||||
<PropertyGroup>
|
<PropertyGroup>
|
||||||
<PostBuildEvent>
|
<PostBuildEvent>
|
||||||
|
@ -201,10 +209,10 @@
|
||||||
<PropertyGroup>
|
<PropertyGroup>
|
||||||
<ErrorText>This project references NuGet package(s) that are missing on this computer. Use NuGet Package Restore to download them. For more information, see http://go.microsoft.com/fwlink/?LinkID=322105. The missing file is {0}.</ErrorText>
|
<ErrorText>This project references NuGet package(s) that are missing on this computer. Use NuGet Package Restore to download them. For more information, see http://go.microsoft.com/fwlink/?LinkID=322105. The missing file is {0}.</ErrorText>
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
<Error Condition="!Exists('..\packages\cef.redist.x64.111.2.2\build\cef.redist.x64.props')" Text="$([System.String]::Format('$(ErrorText)', '..\packages\cef.redist.x64.111.2.2\build\cef.redist.x64.props'))" />
|
<Error Condition="!Exists('..\packages\chromiumembeddedframework.runtime.win-x64.121.3.13\build\chromiumembeddedframework.runtime.win-x64.props')" Text="$([System.String]::Format('$(ErrorText)', '..\packages\chromiumembeddedframework.runtime.win-x64.121.3.13\build\chromiumembeddedframework.runtime.win-x64.props'))" />
|
||||||
<Error Condition="!Exists('..\packages\cef.redist.x86.111.2.2\build\cef.redist.x86.props')" Text="$([System.String]::Format('$(ErrorText)', '..\packages\cef.redist.x86.111.2.2\build\cef.redist.x86.props'))" />
|
<Error Condition="!Exists('..\packages\chromiumembeddedframework.runtime.win-x86.121.3.13\build\chromiumembeddedframework.runtime.win-x86.props')" Text="$([System.String]::Format('$(ErrorText)', '..\packages\chromiumembeddedframework.runtime.win-x86.121.3.13\build\chromiumembeddedframework.runtime.win-x86.props'))" />
|
||||||
<Error Condition="!Exists('..\packages\CefSharp.Common.111.2.20\build\CefSharp.Common.props')" Text="$([System.String]::Format('$(ErrorText)', '..\packages\CefSharp.Common.111.2.20\build\CefSharp.Common.props'))" />
|
<Error Condition="!Exists('..\packages\CefSharp.Common.121.3.130\build\CefSharp.Common.props')" Text="$([System.String]::Format('$(ErrorText)', '..\packages\CefSharp.Common.121.3.130\build\CefSharp.Common.props'))" />
|
||||||
<Error Condition="!Exists('..\packages\CefSharp.Common.111.2.20\build\CefSharp.Common.targets')" Text="$([System.String]::Format('$(ErrorText)', '..\packages\CefSharp.Common.111.2.20\build\CefSharp.Common.targets'))" />
|
<Error Condition="!Exists('..\packages\CefSharp.Common.121.3.130\build\CefSharp.Common.targets')" Text="$([System.String]::Format('$(ErrorText)', '..\packages\CefSharp.Common.121.3.130\build\CefSharp.Common.targets'))" />
|
||||||
</Target>
|
</Target>
|
||||||
<Import Project="..\packages\CefSharp.Common.111.2.20\build\CefSharp.Common.targets" Condition="Exists('..\packages\CefSharp.Common.111.2.20\build\CefSharp.Common.targets')" />
|
<Import Project="..\packages\CefSharp.Common.121.3.130\build\CefSharp.Common.targets" Condition="Exists('..\packages\CefSharp.Common.121.3.130\build\CefSharp.Common.targets')" />
|
||||||
</Project>
|
</Project>
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
||||||
*
|
*
|
||||||
* This Source Code Form is subject to the terms of the Mozilla Public
|
* 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
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
|
Some files were not shown because too many files have changed in this diff Show more
Loading…
Reference in a new issue