2018-12-21 11:36:20 +01:00
|
|
|
|
/*
|
2019-01-09 11:25:21 +01:00
|
|
|
|
* Copyright (c) 2019 ETH Zürich, Educational Development and Technology (LET)
|
2018-12-21 11:36:20 +01: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.Collections.Generic;
|
|
|
|
|
using System.IO;
|
|
|
|
|
using System.Text;
|
|
|
|
|
using SafeExamBrowser.Configuration.Cryptography;
|
|
|
|
|
using SafeExamBrowser.Contracts.Configuration;
|
|
|
|
|
using SafeExamBrowser.Contracts.Configuration.Cryptography;
|
|
|
|
|
using SafeExamBrowser.Contracts.Configuration.DataCompression;
|
|
|
|
|
using SafeExamBrowser.Contracts.Configuration.DataFormats;
|
|
|
|
|
using SafeExamBrowser.Contracts.Logging;
|
|
|
|
|
|
|
|
|
|
namespace SafeExamBrowser.Configuration.DataFormats
|
|
|
|
|
{
|
|
|
|
|
public class BinarySerializer : IDataSerializer
|
|
|
|
|
{
|
|
|
|
|
private IDataCompressor compressor;
|
|
|
|
|
private IModuleLogger logger;
|
|
|
|
|
|
|
|
|
|
public BinarySerializer(IDataCompressor compressor, IModuleLogger logger)
|
|
|
|
|
{
|
|
|
|
|
this.compressor = compressor;
|
|
|
|
|
this.logger = logger;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public bool CanSerialize(FormatType format)
|
|
|
|
|
{
|
|
|
|
|
return format == FormatType.Binary;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public SerializeResult TrySerialize(IDictionary<string, object> data, EncryptionParameters encryption = null)
|
|
|
|
|
{
|
|
|
|
|
var result = new SerializeResult();
|
|
|
|
|
|
|
|
|
|
switch (encryption)
|
|
|
|
|
{
|
|
|
|
|
case PasswordParameters p:
|
|
|
|
|
result = SerializePasswordBlock(data, p);
|
|
|
|
|
break;
|
|
|
|
|
case PublicKeyHashParameters p:
|
|
|
|
|
result = SerializePublicKeyHashBlock(data, p);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
result = SerializePlainDataBlock(data, true);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (result.Status == SaveStatus.Success)
|
|
|
|
|
{
|
|
|
|
|
result.Data = compressor.Compress(result.Data);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private SerializeResult SerializePasswordBlock(IDictionary<string, object> data, PasswordParameters password)
|
|
|
|
|
{
|
|
|
|
|
var result = SerializePlainDataBlock(data);
|
|
|
|
|
|
|
|
|
|
if (result.Status == SaveStatus.Success)
|
|
|
|
|
{
|
|
|
|
|
var encryption = new PasswordEncryption(logger.CloneFor(nameof(PasswordEncryption)));
|
|
|
|
|
var prefix = password.IsHash ? BinaryBlock.PasswordConfigureClient : BinaryBlock.Password;
|
|
|
|
|
|
|
|
|
|
logger.Debug("Attempting to serialize password block...");
|
|
|
|
|
|
|
|
|
|
var status = encryption.Encrypt(result.Data, password.Password, out var encrypted);
|
|
|
|
|
|
|
|
|
|
if (status == SaveStatus.Success)
|
|
|
|
|
{
|
|
|
|
|
result.Data = WritePrefix(prefix, encrypted);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
result.Status = status;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private SerializeResult SerializePlainDataBlock(IDictionary<string, object> data, bool writePrefix = false)
|
|
|
|
|
{
|
|
|
|
|
logger.Debug("Attempting to serialize plain data block...");
|
|
|
|
|
|
|
|
|
|
var xmlSerializer = new XmlSerializer(logger.CloneFor(nameof(XmlSerializer)));
|
|
|
|
|
var result = xmlSerializer.TrySerialize(data);
|
|
|
|
|
|
|
|
|
|
if (result.Status == SaveStatus.Success)
|
|
|
|
|
{
|
|
|
|
|
if (writePrefix)
|
|
|
|
|
{
|
|
|
|
|
result.Data = WritePrefix(BinaryBlock.PlainData, result.Data);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
result.Data = compressor.Compress(result.Data);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private SerializeResult SerializePublicKeyHashBlock(IDictionary<string, object> data, PublicKeyHashParameters parameters)
|
|
|
|
|
{
|
2019-01-08 14:10:45 +01:00
|
|
|
|
var result = SerializePublicKeyHashInnerBlock(data, parameters);
|
2018-12-21 11:36:20 +01:00
|
|
|
|
|
|
|
|
|
if (result.Status == SaveStatus.Success)
|
|
|
|
|
{
|
2019-01-08 14:10:45 +01:00
|
|
|
|
var encryption = DetermineEncryptionForPublicKeyHashBlock(parameters);
|
|
|
|
|
var prefix = parameters.SymmetricEncryption ? BinaryBlock.PublicKeyHashWithSymmetricKey : BinaryBlock.PublicKeyHash;
|
2018-12-21 11:36:20 +01:00
|
|
|
|
|
2019-01-08 14:10:45 +01:00
|
|
|
|
logger.Debug("Attempting to serialize public key hash block...");
|
2018-12-21 11:36:20 +01:00
|
|
|
|
|
2019-01-08 14:10:45 +01:00
|
|
|
|
var status = encryption.Encrypt(result.Data, parameters.Certificate, out var encrypted);
|
2018-12-21 11:36:20 +01:00
|
|
|
|
|
2019-01-08 14:10:45 +01:00
|
|
|
|
if (status == SaveStatus.Success)
|
|
|
|
|
{
|
|
|
|
|
result.Data = WritePrefix(prefix, encrypted);
|
2018-12-21 11:36:20 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private SerializeResult SerializePublicKeyHashInnerBlock(IDictionary<string, object> data, PublicKeyHashParameters parameters)
|
|
|
|
|
{
|
|
|
|
|
if (parameters.InnerEncryption is PasswordParameters password)
|
|
|
|
|
{
|
|
|
|
|
return SerializePasswordBlock(data, password);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return SerializePlainDataBlock(data, true);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private PublicKeyHashEncryption DetermineEncryptionForPublicKeyHashBlock(PublicKeyHashParameters parameters)
|
|
|
|
|
{
|
|
|
|
|
var passwordEncryption = new PasswordEncryption(logger.CloneFor(nameof(PasswordEncryption)));
|
|
|
|
|
|
|
|
|
|
if (parameters.SymmetricEncryption)
|
|
|
|
|
{
|
2019-02-19 10:22:32 +01:00
|
|
|
|
return new PublicKeyHashWithSymmetricKeyEncryption(new CertificateStore(), logger.CloneFor(nameof(PublicKeyHashWithSymmetricKeyEncryption)), passwordEncryption);
|
2018-12-21 11:36:20 +01:00
|
|
|
|
}
|
|
|
|
|
|
2019-02-19 10:22:32 +01:00
|
|
|
|
return new PublicKeyHashEncryption(new CertificateStore(), logger.CloneFor(nameof(PublicKeyHashEncryption)));
|
2018-12-21 11:36:20 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private Stream WritePrefix(string prefix, Stream data)
|
|
|
|
|
{
|
|
|
|
|
var prefixBytes = Encoding.UTF8.GetBytes(prefix);
|
|
|
|
|
var stream = new MemoryStream();
|
|
|
|
|
|
|
|
|
|
stream.Write(prefixBytes, 0, prefixBytes.Length);
|
|
|
|
|
|
|
|
|
|
data.Seek(0, SeekOrigin.Begin);
|
|
|
|
|
data.CopyTo(stream);
|
|
|
|
|
|
|
|
|
|
return stream;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|