2018-11-15 08:45:17 +01:00
|
|
|
|
/*
|
|
|
|
|
* Copyright (c) 2018 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.IO;
|
2018-12-14 09:50:10 +01:00
|
|
|
|
using System.Security.Cryptography.X509Certificates;
|
2018-11-15 08:45:17 +01:00
|
|
|
|
using System.Text;
|
2018-12-14 09:50:10 +01:00
|
|
|
|
using SafeExamBrowser.Configuration.Cryptography;
|
2018-11-15 08:45:17 +01:00
|
|
|
|
using SafeExamBrowser.Contracts.Configuration;
|
2018-12-14 09:50:10 +01:00
|
|
|
|
using SafeExamBrowser.Contracts.Configuration.Cryptography;
|
|
|
|
|
using SafeExamBrowser.Contracts.Configuration.DataCompression;
|
|
|
|
|
using SafeExamBrowser.Contracts.Configuration.DataFormats;
|
2018-11-15 08:45:17 +01:00
|
|
|
|
using SafeExamBrowser.Contracts.Logging;
|
|
|
|
|
|
|
|
|
|
namespace SafeExamBrowser.Configuration.DataFormats
|
|
|
|
|
{
|
2018-11-22 14:36:20 +01:00
|
|
|
|
public partial class BinaryFormat : IDataFormat
|
2018-11-15 08:45:17 +01:00
|
|
|
|
{
|
|
|
|
|
private const int PREFIX_LENGTH = 4;
|
|
|
|
|
|
|
|
|
|
private IDataCompressor compressor;
|
2018-11-28 15:43:30 +01:00
|
|
|
|
private IHashAlgorithm hashAlgorithm;
|
|
|
|
|
private IModuleLogger logger;
|
2018-11-15 08:45:17 +01:00
|
|
|
|
|
2018-11-28 15:43:30 +01:00
|
|
|
|
public BinaryFormat(IDataCompressor compressor, IHashAlgorithm hashAlgorithm, IModuleLogger logger)
|
2018-11-15 08:45:17 +01:00
|
|
|
|
{
|
|
|
|
|
this.compressor = compressor;
|
2018-11-28 15:43:30 +01:00
|
|
|
|
this.hashAlgorithm = hashAlgorithm;
|
2018-11-15 08:45:17 +01:00
|
|
|
|
this.logger = logger;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public bool CanParse(Stream data)
|
|
|
|
|
{
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
var longEnough = data.Length > PREFIX_LENGTH;
|
|
|
|
|
|
|
|
|
|
if (longEnough)
|
|
|
|
|
{
|
|
|
|
|
var prefix = ParsePrefix(data);
|
2018-11-22 14:36:20 +01:00
|
|
|
|
var success = TryDetermineFormat(prefix, out FormatType format);
|
2018-11-15 08:45:17 +01:00
|
|
|
|
|
|
|
|
|
logger.Debug($"'{data}' starting with '{prefix}' does {(success ? string.Empty : "not ")}match the binary format.");
|
|
|
|
|
|
|
|
|
|
return success;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
logger.Debug($"'{data}' is not long enough ({data.Length} bytes) to match the binary format.");
|
|
|
|
|
}
|
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
|
|
|
|
logger.Error($"Failed to determine whether '{data}' with {data.Length / 1000.0} KB data matches the binary format!", e);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2018-12-14 15:30:10 +01:00
|
|
|
|
public ParseResult TryParse(Stream data, PasswordParameters password = null)
|
2018-11-15 08:45:17 +01:00
|
|
|
|
{
|
2018-11-22 14:36:20 +01:00
|
|
|
|
var prefix = ParsePrefix(data);
|
|
|
|
|
var success = TryDetermineFormat(prefix, out FormatType format);
|
2018-11-15 08:45:17 +01:00
|
|
|
|
|
2018-11-22 14:36:20 +01:00
|
|
|
|
if (success)
|
|
|
|
|
{
|
|
|
|
|
if (compressor.IsCompressed(data))
|
|
|
|
|
{
|
|
|
|
|
data = compressor.Decompress(data);
|
|
|
|
|
}
|
|
|
|
|
|
2018-11-28 15:43:30 +01:00
|
|
|
|
logger.Debug($"Attempting to parse '{data}' with format '{prefix}'...");
|
2018-11-30 14:50:28 +01:00
|
|
|
|
data = new SubStream(data, PREFIX_LENGTH, data.Length - PREFIX_LENGTH);
|
2018-11-22 14:36:20 +01:00
|
|
|
|
|
|
|
|
|
switch (format)
|
|
|
|
|
{
|
|
|
|
|
case FormatType.Password:
|
|
|
|
|
case FormatType.PasswordConfigureClient:
|
2018-12-14 09:50:10 +01:00
|
|
|
|
return ParsePasswordBlock(data, format, password);
|
2018-11-22 14:36:20 +01:00
|
|
|
|
case FormatType.PlainData:
|
2018-12-14 09:50:10 +01:00
|
|
|
|
return ParsePlainDataBlock(data);
|
2018-11-22 14:36:20 +01:00
|
|
|
|
case FormatType.PublicKeyHash:
|
2018-12-14 09:50:10 +01:00
|
|
|
|
return ParsePublicKeyHashBlock(data, password);
|
2018-11-28 15:43:30 +01:00
|
|
|
|
case FormatType.PublicKeyHashWithSymmetricKey:
|
2018-12-14 09:50:10 +01:00
|
|
|
|
return ParsePublicKeyHashWithSymmetricKeyBlock(data, password);
|
2018-11-22 14:36:20 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
logger.Error($"'{data}' starting with '{prefix}' does not match the binary format!");
|
|
|
|
|
|
2018-12-11 16:06:10 +01:00
|
|
|
|
return new ParseResult { Status = LoadStatus.InvalidData };
|
2018-11-15 08:45:17 +01:00
|
|
|
|
}
|
|
|
|
|
|
2018-12-14 15:30:10 +01:00
|
|
|
|
private ParseResult ParsePasswordBlock(Stream data, FormatType format, PasswordParameters password = null)
|
2018-11-28 15:43:30 +01:00
|
|
|
|
{
|
2018-11-30 14:50:28 +01:00
|
|
|
|
var encryption = new PasswordEncryption(logger.CloneFor(nameof(PasswordEncryption)));
|
2018-12-14 09:50:10 +01:00
|
|
|
|
var encryptionParams = new PasswordParameters();
|
|
|
|
|
var result = new ParseResult();
|
2018-11-28 15:43:30 +01:00
|
|
|
|
|
2018-12-14 15:30:10 +01:00
|
|
|
|
if (password != null)
|
2018-11-30 14:50:28 +01:00
|
|
|
|
{
|
2018-12-14 15:30:10 +01:00
|
|
|
|
if (format == FormatType.PasswordConfigureClient)
|
|
|
|
|
{
|
|
|
|
|
encryptionParams.Password = password.IsHash ? password.Password : hashAlgorithm.GenerateHashFor(password.Password);
|
|
|
|
|
encryptionParams.IsHash = true;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
encryptionParams.Password = password.Password;
|
|
|
|
|
encryptionParams.IsHash = password.IsHash;
|
|
|
|
|
}
|
2018-11-28 15:43:30 +01:00
|
|
|
|
|
2018-12-14 15:30:10 +01:00
|
|
|
|
result.Status = encryption.Decrypt(data, encryptionParams.Password, out var decrypted);
|
2018-12-14 09:50:10 +01:00
|
|
|
|
|
2018-12-14 15:30:10 +01:00
|
|
|
|
if (result.Status == LoadStatus.Success)
|
|
|
|
|
{
|
|
|
|
|
result = ParsePlainDataBlock(decrypted);
|
|
|
|
|
result.Encryption = encryptionParams;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
2018-11-28 15:43:30 +01:00
|
|
|
|
{
|
2018-12-14 15:30:10 +01:00
|
|
|
|
result.Status = LoadStatus.PasswordNeeded;
|
2018-11-28 15:43:30 +01:00
|
|
|
|
}
|
|
|
|
|
|
2018-12-14 09:50:10 +01:00
|
|
|
|
return result;
|
2018-11-28 15:43:30 +01:00
|
|
|
|
}
|
|
|
|
|
|
2018-12-14 09:50:10 +01:00
|
|
|
|
private ParseResult ParsePlainDataBlock(Stream data)
|
2018-11-28 15:43:30 +01:00
|
|
|
|
{
|
|
|
|
|
var xmlFormat = new XmlFormat(logger.CloneFor(nameof(XmlFormat)));
|
|
|
|
|
|
|
|
|
|
if (compressor.IsCompressed(data))
|
|
|
|
|
{
|
|
|
|
|
data = compressor.Decompress(data);
|
|
|
|
|
}
|
|
|
|
|
|
2018-12-14 09:50:10 +01:00
|
|
|
|
return xmlFormat.TryParse(data);
|
2018-11-28 15:43:30 +01:00
|
|
|
|
}
|
|
|
|
|
|
2018-12-14 15:30:10 +01:00
|
|
|
|
private ParseResult ParsePublicKeyHashBlock(Stream data, PasswordParameters password = null)
|
2018-11-28 15:43:30 +01:00
|
|
|
|
{
|
2018-11-29 15:00:10 +01:00
|
|
|
|
var encryption = new PublicKeyHashEncryption(logger.CloneFor(nameof(PublicKeyHashEncryption)));
|
2018-12-14 09:50:10 +01:00
|
|
|
|
var result = new ParseResult();
|
|
|
|
|
|
|
|
|
|
result.Status = encryption.Decrypt(data, out var decrypted, out var certificate);
|
2018-11-29 15:00:10 +01:00
|
|
|
|
|
2018-12-14 09:50:10 +01:00
|
|
|
|
if (result.Status == LoadStatus.Success)
|
2018-11-29 15:00:10 +01:00
|
|
|
|
{
|
2018-12-14 09:50:10 +01:00
|
|
|
|
result = TryParse(decrypted, password);
|
|
|
|
|
result.Encryption = new PublicKeyHashParameters
|
|
|
|
|
{
|
|
|
|
|
Certificate = certificate,
|
|
|
|
|
InnerEncryption = result.Encryption as PasswordParameters,
|
|
|
|
|
SymmetricEncryption = false
|
|
|
|
|
};
|
2018-11-29 15:00:10 +01:00
|
|
|
|
}
|
|
|
|
|
|
2018-12-14 09:50:10 +01:00
|
|
|
|
return result;
|
2018-11-28 15:43:30 +01:00
|
|
|
|
}
|
|
|
|
|
|
2018-12-14 15:30:10 +01:00
|
|
|
|
private ParseResult ParsePublicKeyHashWithSymmetricKeyBlock(Stream data, PasswordParameters password = null)
|
2018-11-28 15:43:30 +01:00
|
|
|
|
{
|
2018-11-29 15:00:10 +01:00
|
|
|
|
var passwordEncryption = new PasswordEncryption(logger.CloneFor(nameof(PasswordEncryption)));
|
2018-12-14 09:50:10 +01:00
|
|
|
|
var encryption = new PublicKeyHashWithSymmetricKeyEncryption(logger.CloneFor(nameof(PublicKeyHashWithSymmetricKeyEncryption)), passwordEncryption);
|
|
|
|
|
var result = new ParseResult();
|
2018-11-29 15:00:10 +01:00
|
|
|
|
|
2018-12-14 09:50:10 +01:00
|
|
|
|
result.Status = encryption.Decrypt(data, out Stream decrypted, out X509Certificate2 certificate);
|
|
|
|
|
|
|
|
|
|
if (result.Status == LoadStatus.Success)
|
2018-11-29 15:00:10 +01:00
|
|
|
|
{
|
2018-12-14 09:50:10 +01:00
|
|
|
|
result = TryParse(decrypted, password);
|
|
|
|
|
result.Encryption = new PublicKeyHashParameters
|
|
|
|
|
{
|
|
|
|
|
Certificate = certificate,
|
|
|
|
|
InnerEncryption = result.Encryption as PasswordParameters,
|
|
|
|
|
SymmetricEncryption = true
|
|
|
|
|
};
|
2018-11-29 15:00:10 +01:00
|
|
|
|
}
|
|
|
|
|
|
2018-12-14 09:50:10 +01:00
|
|
|
|
return result;
|
2018-11-28 15:43:30 +01:00
|
|
|
|
}
|
|
|
|
|
|
2018-11-15 08:45:17 +01:00
|
|
|
|
private string ParsePrefix(Stream data)
|
|
|
|
|
{
|
|
|
|
|
var prefixData = new byte[PREFIX_LENGTH];
|
|
|
|
|
|
|
|
|
|
if (compressor.IsCompressed(data))
|
|
|
|
|
{
|
|
|
|
|
prefixData = compressor.Peek(data, PREFIX_LENGTH);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
data.Seek(0, SeekOrigin.Begin);
|
|
|
|
|
data.Read(prefixData, 0, PREFIX_LENGTH);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return Encoding.UTF8.GetString(prefixData);
|
|
|
|
|
}
|
|
|
|
|
|
2018-11-22 14:36:20 +01:00
|
|
|
|
private bool TryDetermineFormat(string prefix, out FormatType format)
|
2018-11-15 08:45:17 +01:00
|
|
|
|
{
|
2018-11-22 14:36:20 +01:00
|
|
|
|
format = default(FormatType);
|
2018-11-15 08:45:17 +01:00
|
|
|
|
|
|
|
|
|
switch (prefix)
|
|
|
|
|
{
|
|
|
|
|
case "pswd":
|
2018-11-22 14:36:20 +01:00
|
|
|
|
format = FormatType.Password;
|
2018-11-15 08:45:17 +01:00
|
|
|
|
return true;
|
|
|
|
|
case "pwcc":
|
2018-11-22 14:36:20 +01:00
|
|
|
|
format = FormatType.PasswordConfigureClient;
|
2018-11-15 08:45:17 +01:00
|
|
|
|
return true;
|
|
|
|
|
case "plnd":
|
2018-11-22 14:36:20 +01:00
|
|
|
|
format = FormatType.PlainData;
|
2018-11-15 08:45:17 +01:00
|
|
|
|
return true;
|
|
|
|
|
case "pkhs":
|
2018-11-22 14:36:20 +01:00
|
|
|
|
format = FormatType.PublicKeyHash;
|
2018-11-15 08:45:17 +01:00
|
|
|
|
return true;
|
|
|
|
|
case "phsk":
|
2018-11-28 15:43:30 +01:00
|
|
|
|
format = FormatType.PublicKeyHashWithSymmetricKey;
|
2018-11-15 08:45:17 +01:00
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2018-11-22 14:36:20 +01:00
|
|
|
|
private enum FormatType
|
2018-11-15 08:45:17 +01:00
|
|
|
|
{
|
|
|
|
|
Password = 1,
|
2018-11-22 14:36:20 +01:00
|
|
|
|
PasswordConfigureClient,
|
2018-11-15 08:45:17 +01:00
|
|
|
|
PlainData,
|
|
|
|
|
PublicKeyHash,
|
2018-11-28 15:43:30 +01:00
|
|
|
|
PublicKeyHashWithSymmetricKey
|
2018-11-15 08:45:17 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|