2019-09-10 11:01:49 +02:00
|
|
|
|
/*
|
|
|
|
|
* Copyright (c) 2019 ETH Zürich, Educational Development and Technology (LET)
|
|
|
|
|
*
|
|
|
|
|
* 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.Specialized;
|
|
|
|
|
using System.IO;
|
|
|
|
|
using System.Reflection;
|
|
|
|
|
using CefSharp;
|
|
|
|
|
using SafeExamBrowser.Browser.Filters;
|
|
|
|
|
using SafeExamBrowser.Configuration.Contracts;
|
|
|
|
|
using SafeExamBrowser.I18n.Contracts;
|
|
|
|
|
using SafeExamBrowser.Logging.Contracts;
|
|
|
|
|
using SafeExamBrowser.Settings.Browser;
|
|
|
|
|
|
|
|
|
|
namespace SafeExamBrowser.Browser.Handlers
|
|
|
|
|
{
|
|
|
|
|
internal class ResourceHandler : CefSharp.Handler.ResourceRequestHandler
|
|
|
|
|
{
|
|
|
|
|
private AppConfig appConfig;
|
|
|
|
|
private BrowserFilterSettings settings;
|
|
|
|
|
private ILogger logger;
|
|
|
|
|
private RequestFilter filter;
|
2019-09-12 12:31:17 +02:00
|
|
|
|
private IResourceHandler contentHandler;
|
|
|
|
|
private IResourceHandler pageHandler;
|
2019-09-10 11:01:49 +02:00
|
|
|
|
private IText text;
|
|
|
|
|
|
2019-09-12 12:31:17 +02:00
|
|
|
|
internal ResourceHandler(AppConfig appConfig, BrowserFilterSettings settings, RequestFilter filter, ILogger logger, IText text)
|
2019-09-10 11:01:49 +02:00
|
|
|
|
{
|
|
|
|
|
this.appConfig = appConfig;
|
2019-09-12 12:31:17 +02:00
|
|
|
|
this.filter = filter;
|
2019-09-10 11:01:49 +02:00
|
|
|
|
this.logger = logger;
|
|
|
|
|
this.settings = settings;
|
|
|
|
|
this.text = text;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
internal void Initialize()
|
|
|
|
|
{
|
2019-09-12 12:31:17 +02:00
|
|
|
|
if (settings.FilterContentRequests)
|
2019-09-10 11:01:49 +02:00
|
|
|
|
{
|
2019-09-12 12:31:17 +02:00
|
|
|
|
InitializeResourceHandlers();
|
2019-09-10 11:01:49 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected override IResourceHandler GetResourceHandler(IWebBrowser chromiumWebBrowser, IBrowser browser, IFrame frame, IRequest request)
|
|
|
|
|
{
|
2019-09-12 12:31:17 +02:00
|
|
|
|
if (Block(request))
|
2019-09-10 11:01:49 +02:00
|
|
|
|
{
|
2019-09-12 12:31:17 +02:00
|
|
|
|
return ResourceHandlerFor(request.ResourceType);
|
2019-09-10 11:01:49 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return base.GetResourceHandler(chromiumWebBrowser, browser, frame, request);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected override CefReturnValue OnBeforeResourceLoad(IWebBrowser webBrowser, IBrowser browser, IFrame frame, IRequest request, IRequestCallback callback)
|
|
|
|
|
{
|
|
|
|
|
// TODO: CEF does not yet support intercepting requests from service workers, thus the user agent must be statically set at browser
|
|
|
|
|
// startup for now. Once CEF has full support of service workers, the static user agent should be removed and the method below
|
|
|
|
|
// reactivated. See https://bitbucket.org/chromiumembedded/cef/issues/2622 for the current status of development.
|
|
|
|
|
// AppendCustomUserAgent(request);
|
|
|
|
|
|
|
|
|
|
if (IsMailtoUrl(request.Url))
|
|
|
|
|
{
|
|
|
|
|
return CefReturnValue.Cancel;
|
|
|
|
|
}
|
|
|
|
|
|
2019-09-12 12:31:17 +02:00
|
|
|
|
ReplaceSebScheme(request);
|
2019-09-10 11:01:49 +02:00
|
|
|
|
|
|
|
|
|
return base.OnBeforeResourceLoad(webBrowser, browser, frame, request, callback);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void AppendCustomUserAgent(IRequest request)
|
|
|
|
|
{
|
|
|
|
|
var headers = new NameValueCollection(request.Headers);
|
|
|
|
|
var userAgent = request.Headers["User-Agent"];
|
|
|
|
|
|
|
|
|
|
headers["User-Agent"] = $"{userAgent} SEB/{appConfig.ProgramInformationalVersion}";
|
|
|
|
|
request.Headers = headers;
|
|
|
|
|
}
|
|
|
|
|
|
2019-09-12 12:31:17 +02:00
|
|
|
|
private bool Block(IRequest request)
|
2019-09-10 11:01:49 +02:00
|
|
|
|
{
|
2019-09-12 12:31:17 +02:00
|
|
|
|
if (settings.FilterContentRequests)
|
2019-09-10 11:01:49 +02:00
|
|
|
|
{
|
|
|
|
|
var result = filter.Process(request.Url);
|
|
|
|
|
var block = result == FilterResult.Block;
|
|
|
|
|
|
|
|
|
|
if (block)
|
|
|
|
|
{
|
|
|
|
|
logger.Info($"Blocked content request for '{request.Url}'.");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return block;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2019-09-12 12:31:17 +02:00
|
|
|
|
private void InitializeResourceHandlers()
|
2019-09-10 11:01:49 +02:00
|
|
|
|
{
|
|
|
|
|
var assembly = Assembly.GetAssembly(typeof(RequestFilter));
|
|
|
|
|
var contentMessage = text.Get(TextKey.Browser_BlockedContentMessage);
|
|
|
|
|
var contentStream = assembly.GetManifestResourceStream($"{typeof(RequestFilter).Namespace}.BlockedContent.html");
|
|
|
|
|
var pageButton = text.Get(TextKey.Browser_BlockedPageButton);
|
|
|
|
|
var pageMessage = text.Get(TextKey.Browser_BlockedPageMessage);
|
|
|
|
|
var pageTitle = text.Get(TextKey.Browser_BlockedPageTitle);
|
|
|
|
|
var pageStream = assembly.GetManifestResourceStream($"{typeof(RequestFilter).Namespace}.BlockedPage.html");
|
|
|
|
|
var contentHtml = new StreamReader(contentStream).ReadToEnd();
|
|
|
|
|
var pageHtml = new StreamReader(pageStream).ReadToEnd();
|
|
|
|
|
|
|
|
|
|
contentHtml = contentHtml.Replace("%%MESSAGE%%", contentMessage);
|
2019-09-12 12:31:17 +02:00
|
|
|
|
contentHandler = CefSharp.ResourceHandler.FromString(contentHtml);
|
2019-09-10 11:01:49 +02:00
|
|
|
|
|
2019-09-12 12:31:17 +02:00
|
|
|
|
pageHtml = pageHtml.Replace("%%MESSAGE%%", pageMessage).Replace("%%TITLE%%", pageTitle).Replace("%%BACK_BUTTON%%", pageButton);
|
|
|
|
|
pageHandler = CefSharp.ResourceHandler.FromString(pageHtml);
|
2019-09-10 11:01:49 +02:00
|
|
|
|
|
2019-09-12 12:31:17 +02:00
|
|
|
|
logger.Debug("Initialized resource handlers for blocked requests.");
|
2019-09-10 11:01:49 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private bool IsMailtoUrl(string url)
|
|
|
|
|
{
|
|
|
|
|
return url.StartsWith(Uri.UriSchemeMailto);
|
|
|
|
|
}
|
|
|
|
|
|
2019-09-12 12:31:17 +02:00
|
|
|
|
private void ReplaceSebScheme(IRequest request)
|
2019-09-10 11:01:49 +02:00
|
|
|
|
{
|
|
|
|
|
if (Uri.IsWellFormedUriString(request.Url, UriKind.RelativeOrAbsolute))
|
|
|
|
|
{
|
|
|
|
|
var uri = new Uri(request.Url);
|
|
|
|
|
|
|
|
|
|
if (uri.Scheme == appConfig.SebUriScheme)
|
|
|
|
|
{
|
|
|
|
|
request.Url = new UriBuilder(uri) { Scheme = Uri.UriSchemeHttp }.Uri.AbsoluteUri;
|
|
|
|
|
}
|
|
|
|
|
else if (uri.Scheme == appConfig.SebUriSchemeSecure)
|
|
|
|
|
{
|
|
|
|
|
request.Url = new UriBuilder(uri) { Scheme = Uri.UriSchemeHttps }.Uri.AbsoluteUri;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2019-09-12 12:31:17 +02:00
|
|
|
|
|
|
|
|
|
private IResourceHandler ResourceHandlerFor(ResourceType resourceType)
|
|
|
|
|
{
|
|
|
|
|
switch (resourceType)
|
|
|
|
|
{
|
|
|
|
|
case ResourceType.MainFrame:
|
|
|
|
|
case ResourceType.SubFrame:
|
|
|
|
|
return pageHandler;
|
|
|
|
|
default:
|
|
|
|
|
return contentHandler;
|
|
|
|
|
}
|
|
|
|
|
}
|
2019-09-10 11:01:49 +02:00
|
|
|
|
}
|
|
|
|
|
}
|