/* * 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.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 data, EncryptionParameters encryption = null) { var result = new SerializeResult(); switch (encryption) { case PasswordParameters p: result = SerializePasswordBlock(data, p); break; case PublicKeyParameters 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 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 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 data, PublicKeyParameters parameters) { var result = SerializePublicKeyHashInnerBlock(data, parameters); if (result.Status == SaveStatus.Success) { var encryption = DetermineEncryptionForPublicKeyHashBlock(parameters); var prefix = parameters.SymmetricEncryption ? BinaryBlock.PublicKeySymmetric : BinaryBlock.PublicKey; logger.Debug("Attempting to serialize public key hash block..."); var status = encryption.Encrypt(result.Data, parameters.Certificate, out var encrypted); if (status == SaveStatus.Success) { result.Data = WritePrefix(prefix, encrypted); } } return result; } private SerializeResult SerializePublicKeyHashInnerBlock(IDictionary data, PublicKeyParameters parameters) { if (parameters.InnerEncryption is PasswordParameters password) { return SerializePasswordBlock(data, password); } return SerializePlainDataBlock(data, true); } private PublicKeyEncryption DetermineEncryptionForPublicKeyHashBlock(PublicKeyParameters parameters) { var passwordEncryption = new PasswordEncryption(logger.CloneFor(nameof(PasswordEncryption))); if (parameters.SymmetricEncryption) { return new PublicKeySymmetricEncryption(new CertificateStore(), logger.CloneFor(nameof(PublicKeySymmetricEncryption)), passwordEncryption); } return new PublicKeyEncryption(new CertificateStore(), logger.CloneFor(nameof(PublicKeyEncryption))); } 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; } } }