2018-06-21 07:56:25 +02:00
|
|
|
|
/*
|
2019-01-09 11:25:21 +01:00
|
|
|
|
* Copyright (c) 2019 ETH Zürich, Educational Development and Technology (LET)
|
2018-06-21 07:56:25 +02:00
|
|
|
|
*
|
|
|
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Concurrent;
|
|
|
|
|
using System.IO;
|
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
|
using CefSharp;
|
2019-09-06 08:13:27 +02:00
|
|
|
|
using SafeExamBrowser.Browser.Contracts.Events;
|
2019-08-30 09:55:26 +02:00
|
|
|
|
using SafeExamBrowser.Configuration.Contracts;
|
|
|
|
|
using SafeExamBrowser.Logging.Contracts;
|
2019-09-06 09:39:28 +02:00
|
|
|
|
using BrowserSettings = SafeExamBrowser.Settings.Browser.BrowserSettings;
|
2018-06-21 07:56:25 +02:00
|
|
|
|
|
|
|
|
|
namespace SafeExamBrowser.Browser.Handlers
|
|
|
|
|
{
|
|
|
|
|
internal class DownloadHandler : IDownloadHandler
|
|
|
|
|
{
|
2018-06-29 09:50:20 +02:00
|
|
|
|
private AppConfig appConfig;
|
2018-06-21 07:56:25 +02:00
|
|
|
|
private BrowserSettings settings;
|
|
|
|
|
private ConcurrentDictionary<int, DownloadFinishedCallback> callbacks;
|
2018-08-31 15:29:36 +02:00
|
|
|
|
private ILogger logger;
|
2018-06-21 07:56:25 +02:00
|
|
|
|
|
|
|
|
|
public event DownloadRequestedEventHandler ConfigurationDownloadRequested;
|
|
|
|
|
|
2018-08-31 15:29:36 +02:00
|
|
|
|
public DownloadHandler(AppConfig appConfig, BrowserSettings settings, ILogger logger)
|
2018-06-21 07:56:25 +02:00
|
|
|
|
{
|
2018-06-29 09:50:20 +02:00
|
|
|
|
this.appConfig = appConfig;
|
2018-06-21 07:56:25 +02:00
|
|
|
|
this.callbacks = new ConcurrentDictionary<int, DownloadFinishedCallback>();
|
2018-08-31 15:29:36 +02:00
|
|
|
|
this.logger = logger;
|
2018-06-21 07:56:25 +02:00
|
|
|
|
this.settings = settings;
|
|
|
|
|
}
|
|
|
|
|
|
2019-01-08 14:30:38 +01:00
|
|
|
|
public void OnBeforeDownload(IWebBrowser webBrowser, IBrowser browser, DownloadItem downloadItem, IBeforeDownloadCallback callback)
|
2018-06-21 07:56:25 +02:00
|
|
|
|
{
|
|
|
|
|
var uri = new Uri(downloadItem.Url);
|
|
|
|
|
var extension = Path.GetExtension(uri.AbsolutePath);
|
2018-12-11 16:06:10 +01:00
|
|
|
|
var isConfigFile = String.Equals(extension, appConfig.ConfigurationFileExtension, StringComparison.OrdinalIgnoreCase);
|
2018-06-21 07:56:25 +02:00
|
|
|
|
|
2018-08-31 15:29:36 +02:00
|
|
|
|
logger.Debug($"Handling download request for '{uri}'.");
|
|
|
|
|
|
2018-06-21 07:56:25 +02:00
|
|
|
|
if (isConfigFile)
|
|
|
|
|
{
|
|
|
|
|
Task.Run(() => RequestConfigurationFileDownload(downloadItem, callback));
|
|
|
|
|
}
|
2018-08-31 15:29:36 +02:00
|
|
|
|
else if (settings.AllowDownloads)
|
2018-06-21 07:56:25 +02:00
|
|
|
|
{
|
2018-08-31 15:29:36 +02:00
|
|
|
|
logger.Debug($"Starting download of '{uri}'...");
|
|
|
|
|
|
2018-06-21 07:56:25 +02:00
|
|
|
|
using (callback)
|
|
|
|
|
{
|
|
|
|
|
callback.Continue(null, true);
|
|
|
|
|
}
|
|
|
|
|
}
|
2018-08-31 15:29:36 +02:00
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
logger.Info($"Aborted download request for '{uri}', as downloading is not allowed.");
|
|
|
|
|
}
|
2018-06-21 07:56:25 +02:00
|
|
|
|
}
|
|
|
|
|
|
2019-01-08 14:30:38 +01:00
|
|
|
|
public void OnDownloadUpdated(IWebBrowser webBrowser, IBrowser browser, DownloadItem downloadItem, IDownloadItemCallback callback)
|
2018-06-21 07:56:25 +02:00
|
|
|
|
{
|
|
|
|
|
if (downloadItem.IsComplete || downloadItem.IsCancelled)
|
|
|
|
|
{
|
|
|
|
|
if (callbacks.TryRemove(downloadItem.Id, out DownloadFinishedCallback finished) && finished != null)
|
|
|
|
|
{
|
|
|
|
|
Task.Run(() => finished.Invoke(downloadItem.IsComplete, downloadItem.FullPath));
|
|
|
|
|
}
|
2018-08-31 15:29:36 +02:00
|
|
|
|
|
|
|
|
|
logger.Debug($"Download of '{downloadItem.Url}' {(downloadItem.IsComplete ? "is complete" : "was cancelled")}.");
|
2018-06-21 07:56:25 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void RequestConfigurationFileDownload(DownloadItem downloadItem, IBeforeDownloadCallback callback)
|
|
|
|
|
{
|
|
|
|
|
var args = new DownloadEventArgs();
|
|
|
|
|
|
2018-10-30 11:24:28 +01:00
|
|
|
|
logger.Debug($"Detected download request for configuration file '{downloadItem.Url}'.");
|
2018-06-21 07:56:25 +02:00
|
|
|
|
ConfigurationDownloadRequested?.Invoke(downloadItem.SuggestedFileName, args);
|
|
|
|
|
|
|
|
|
|
if (args.AllowDownload)
|
|
|
|
|
{
|
|
|
|
|
if (args.Callback != null)
|
|
|
|
|
{
|
|
|
|
|
callbacks[downloadItem.Id] = args.Callback;
|
|
|
|
|
}
|
|
|
|
|
|
2019-09-04 15:12:59 +02:00
|
|
|
|
logger.Debug($"Starting download of configuration file '{downloadItem.Url}'...");
|
|
|
|
|
|
2018-06-21 07:56:25 +02:00
|
|
|
|
using (callback)
|
|
|
|
|
{
|
|
|
|
|
callback.Continue(args.DownloadPath, false);
|
|
|
|
|
}
|
|
|
|
|
}
|
2019-09-04 15:12:59 +02:00
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
logger.Debug($"Download of configuration file '{downloadItem.Url}' was cancelled.");
|
|
|
|
|
}
|
2018-06-21 07:56:25 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|