#region License Information
/* HeuristicLab
* Copyright (C) 2002-2018 Heuristic and Evolutionary Algorithms Laboratory (HEAL)
*
* This file is part of HeuristicLab.
*
* HeuristicLab is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* HeuristicLab is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with HeuristicLab. If not, see .
*/
#endregion
using System;
using System.Collections.Generic;
using System.IO;
using System.IO.Compression;
using System.Linq;
using System.Text;
using System.Threading;
using HeuristicLab.Persistence.Core;
using HeuristicLab.Persistence.Core.Tokens;
using HeuristicLab.Persistence.Interfaces;
using HeuristicLab.Tracing;
namespace HeuristicLab.Persistence.Default.Xml {
///
/// Main entry point of persistence to XML. Use the static methods to serialize
/// to a file or to a stream.
///
public class XmlGenerator : GeneratorBase {
protected int depth;
protected int Depth {
get {
return depth;
}
set {
depth = value;
prefix = new string(' ', depth * 2);
}
}
protected string prefix;
///
/// Initializes a new instance of the class.
///
public XmlGenerator() {
Depth = 0;
}
protected enum NodeType { Start, End, Inline };
protected static void AddXmlTagContent(StringBuilder sb, string name, Dictionary attributes) {
sb.Append(name);
foreach (var attribute in attributes) {
if (attribute.Value != null && !string.IsNullOrEmpty(attribute.Value.ToString())) {
sb.Append(' ');
sb.Append(attribute.Key);
sb.Append("=\"");
sb.Append(attribute.Value);
sb.Append('"');
}
}
}
protected static int AttributeLength(Dictionary attributes) {
return attributes
.Where(kvp => !string.IsNullOrEmpty(kvp.Key) && !string.IsNullOrEmpty(kvp.Value))
.Select(kvp => kvp.Key.Length + kvp.Value.Length + 4).Sum();
}
protected static void AddXmlStartTag(StringBuilder sb, string name, Dictionary attributes) {
sb.Append('<');
AddXmlTagContent(sb, name, attributes);
sb.Append('>');
}
protected static void AddXmlInlineTag(StringBuilder sb, string name, Dictionary attributes) {
sb.Append('<');
AddXmlTagContent(sb, name, attributes);
sb.Append("/>");
}
protected static void AddXmlEndTag(StringBuilder sb, string name) {
sb.Append("");
sb.Append(name);
sb.Append(">");
}
protected string CreateNodeStart(string name, Dictionary attributes) {
StringBuilder sb = new StringBuilder(prefix.Length + name.Length + 4
+ AttributeLength(attributes));
sb.Append(prefix);
Depth += 1;
AddXmlStartTag(sb, name, attributes);
sb.Append("\r\n");
return sb.ToString();
}
private static Dictionary emptyDict = new Dictionary();
protected string CreateNodeStart(string name) {
return CreateNodeStart(name, emptyDict);
}
protected string CreateNodeEnd(string name) {
Depth -= 1;
StringBuilder sb = new StringBuilder(prefix.Length + name.Length + 5);
sb.Append(prefix);
AddXmlEndTag(sb, name);
sb.Append("\r\n");
return sb.ToString();
}
protected string CreateNode(string name, Dictionary attributes) {
StringBuilder sb = new StringBuilder(prefix.Length + name.Length + 5
+ AttributeLength(attributes));
sb.Append(prefix);
AddXmlInlineTag(sb, name, attributes);
sb.Append("\r\n");
return sb.ToString();
}
protected string CreateNode(string name, Dictionary attributes, string content) {
StringBuilder sb = new StringBuilder(
prefix.Length + name.Length + AttributeLength(attributes) + 2
+ content.Length + name.Length + 5);
sb.Append(prefix);
AddXmlStartTag(sb, name, attributes);
sb.Append(content);
sb.Append("").Append(name).Append(">\r\n");
return sb.ToString();
}
///
/// Formats the specified begin token.
///
/// The begin token.
/// The token in serialized form.
protected override string Format(BeginToken beginToken) {
var dict = new Dictionary {
{"name", beginToken.Name},
{"typeId", beginToken.TypeId.ToString()},
{"id", beginToken.Id.ToString()}};
AddTypeInfo(beginToken.TypeId, dict);
return CreateNodeStart(XmlStringConstants.COMPOSITE, dict);
}
protected void AddTypeInfo(int typeId, Dictionary dict) {
if (lastTypeToken != null) {
if (typeId == lastTypeToken.Id) {
dict.Add("typeName", lastTypeToken.TypeName);
dict.Add("serializer", lastTypeToken.Serializer);
lastTypeToken = null;
} else {
FlushTypeToken();
}
}
}
///
/// Formats the specified end token.
///
/// The end token.
/// The token in serialized form.
protected override string Format(EndToken endToken) {
return CreateNodeEnd(XmlStringConstants.COMPOSITE);
}
///
/// Formats the specified data token.
///
/// The data token.
/// The token in serialized form.
protected override string Format(PrimitiveToken dataToken) {
var dict = new Dictionary {
{"typeId", dataToken.TypeId.ToString()},
{"name", dataToken.Name},
{"id", dataToken.Id.ToString()}};
AddTypeInfo(dataToken.TypeId, dict);
return CreateNode(XmlStringConstants.PRIMITIVE, dict,
((XmlString)dataToken.SerialData).Data);
}
///
/// Formats the specified ref token.
///
/// The ref token.
/// The token in serialized form.
protected override string Format(ReferenceToken refToken) {
return CreateNode(XmlStringConstants.REFERENCE,
new Dictionary {
{"ref", refToken.Id.ToString()},
{"name", refToken.Name}});
}
///
/// Formats the specified null ref token.
///
/// The null ref token.
/// The token in serialized form.
protected override string Format(NullReferenceToken nullRefToken) {
return CreateNode(XmlStringConstants.NULL,
new Dictionary{
{"name", nullRefToken.Name}});
}
///
/// Formats the specified meta info begin token.
///
/// The meta info begin token.
/// The token in serialized form.
protected override string Format(MetaInfoBeginToken metaInfoBeginToken) {
return CreateNodeStart(XmlStringConstants.METAINFO);
}
///
/// Formats the specified meta info end token.
///
/// The meta info end token.
/// The token in serialized form.
protected override string Format(MetaInfoEndToken metaInfoEndToken) {
return CreateNodeEnd(XmlStringConstants.METAINFO);
}
protected TypeToken lastTypeToken;
///
/// Formats the specified token.
///
/// The token.
/// The token in serialized form.
protected override string Format(TypeToken token) {
lastTypeToken = token;
return "";
}
protected string FlushTypeToken() {
if (lastTypeToken == null)
return "";
try {
return CreateNode(XmlStringConstants.TYPE,
new Dictionary {
{"id", lastTypeToken.Id.ToString()},
{"typeName", lastTypeToken.TypeName },
{"serializer", lastTypeToken.Serializer }});
} finally {
lastTypeToken = null;
}
}
///
/// Formats the specified type cache.
///
/// The type cache.
/// An enumerable of formatted type cache tags.
public IEnumerable Format(List typeCache) {
yield return CreateNodeStart(XmlStringConstants.TYPECACHE);
foreach (var mapping in typeCache)
yield return CreateNode(
XmlStringConstants.TYPE,
mapping.GetDict());
yield return CreateNodeEnd(XmlStringConstants.TYPECACHE);
}
///
/// Serialize an object into a file.
/// The XML configuration is obtained from the ConfigurationService.
/// The file is actually a ZIP file.
/// Compression level is set to 5 and needed assemblies are not included.
///
/// The object.
/// The filename.
public static void Serialize(object o, string filename, CancellationToken cancellationToken = default(CancellationToken)) {
Serialize(o, filename, ConfigurationService.Instance.GetConfiguration(new XmlFormat()), false, CompressionLevel.Optimal, cancellationToken);
}
///
/// Serialize an object into a file.
/// The XML configuration is obtained from the ConfigurationService.
/// Needed assemblies are not included.
///
/// The object.
/// The filename.
/// ZIP file compression level
public static void Serialize(object o, string filename, CompressionLevel compression, CancellationToken cancellationToken = default(CancellationToken)) {
Serialize(o, filename, ConfigurationService.Instance.GetConfiguration(new XmlFormat()), false, compression, cancellationToken);
}
///
/// Serializes the specified object into a file.
/// Needed assemblies are not included, ZIP compression level is set to 5.
///
/// The object.
/// The filename.
/// The configuration.
public static void Serialize(object obj, string filename, Configuration config, CancellationToken cancellationToken = default(CancellationToken)) {
Serialize(obj, filename, config, false, CompressionLevel.Optimal, cancellationToken);
}
private static void Serialize(object obj, Stream stream, Configuration config, bool includeAssemblies, CompressionLevel compression, CancellationToken cancellationToken = default(CancellationToken)) {
Serializer serializer = new Serializer(obj, config);
Serialize(stream, includeAssemblies, compression, serializer, cancellationToken);
}
private static void Serialize(Stream stream, bool includeAssemblies, CompressionLevel compression, Serializer serializer, CancellationToken cancellationToken = default(CancellationToken)) {
try {
cancellationToken.ThrowIfCancellationRequested();
DateTime start = DateTime.Now;
serializer.InterleaveTypeInformation = false;
XmlGenerator generator = new XmlGenerator();
using (ZipArchive zipArchive = new ZipArchive(stream, ZipArchiveMode.Create)) {
ZipArchiveEntry entry = zipArchive.CreateEntry("data.xml", compression);
using (StreamWriter writer = new StreamWriter(entry.Open())) {
foreach (ISerializationToken token in serializer) {
cancellationToken.ThrowIfCancellationRequested();
string line = generator.Format(token);
writer.Write(line);
}
}
entry = zipArchive.CreateEntry("typecache.xml", compression);
using (StreamWriter writer = new StreamWriter(entry.Open())) {
foreach (string line in generator.Format(serializer.TypeCache)) {
writer.Write(line);
}
}
if (includeAssemblies) {
foreach (string name in serializer.RequiredFiles) {
cancellationToken.ThrowIfCancellationRequested();
Uri uri = new Uri(name);
if (!uri.IsFile) {
Logger.Warn("cannot read non-local files");
continue;
}
entry = zipArchive.CreateEntry(Path.GetFileName(uri.PathAndQuery), compression);
using (BinaryWriter bw = new BinaryWriter(entry.Open())) {
using (FileStream reader = File.OpenRead(uri.PathAndQuery)) {
byte[] buffer = new byte[1024 * 1024];
while (true) {
cancellationToken.ThrowIfCancellationRequested();
int bytesRead = reader.Read(buffer, 0, 1024 * 1024);
if (bytesRead == 0)
break;
bw.Write(buffer, 0, bytesRead);
}
}
}
}
}
}
Logger.Info(String.Format("serialization took {0} seconds with compression level {1}",
(DateTime.Now - start).TotalSeconds, compression));
} catch (Exception) {
Logger.Warn("Exception caught, no data has been serialized.");
throw;
}
}
///
/// Serializes the specified object into a file.
///
/// The object.
/// The filename.
/// The configuration.
/// if set to true include needed assemblies.
/// The ZIP compression level.
public static void Serialize(object obj, string filename, Configuration config, bool includeAssemblies, CompressionLevel compression, CancellationToken cancellationToken = default(CancellationToken)) {
string tempfile = null;
try {
tempfile = Path.GetTempFileName();
using (FileStream stream = File.Create(tempfile)) {
Serialize(obj, stream, config, includeAssemblies, compression, cancellationToken);
}
// copy only if needed
File.Copy(tempfile, filename, true);
} catch (Exception) {
Logger.Warn("Exception caught, no data has been written.");
throw;
} finally {
if (tempfile != null && File.Exists(tempfile))
File.Delete(tempfile);
}
}
///
/// Serializes the specified object into a stream using the
/// obtained from the .
///
/// The object.
/// The stream.
/// Type of compression, default is GZip.
public static void Serialize(object obj, Stream stream, CompressionType compressionType = CompressionType.GZip, CancellationToken cancellationToken = default(CancellationToken)) {
Serialize(obj, stream, ConfigurationService.Instance.GetConfiguration(new XmlFormat()), compressionType, cancellationToken);
}
///
/// Serializes the specified object into a stream.
///
/// The object.
/// The stream.
/// The configuration.
/// Type of compression, default is GZip.
public static void Serialize(object obj, Stream stream, Configuration config, CompressionType compressionType = CompressionType.GZip, CancellationToken cancellationToken = default(CancellationToken)) {
Serialize(obj, stream, config, false, compressionType, cancellationToken);
}
///
/// Serializes the specified object into a stream.
///
/// The object.
/// The stream.
/// The configuration.
/// if set to true include need assemblies.
/// Type of compression, default is GZip.
public static void Serialize(object obj, Stream stream, Configuration config, bool includeAssemblies, CompressionType compressionType = CompressionType.GZip, CancellationToken cancellationToken = default(CancellationToken)) {
try {
Serializer serializer = new Serializer(obj, config);
if (compressionType == CompressionType.Zip) {
Serialize(obj, stream, config, includeAssemblies, CompressionLevel.Optimal, cancellationToken);
} else {
Serialize(stream, serializer, cancellationToken);
}
} catch (PersistenceException) {
throw;
} catch (Exception e) {
throw new PersistenceException("Unexpected exception during Serialization.", e);
}
}
///
/// Serializes the specified object into a stream.
///
/// The object.
/// The stream.
/// The configuration.
/// if set to true include need assemblies.
/// The list of all serialized types.
/// Type of compression, default is GZip.
public static void Serialize(object obj, Stream stream, Configuration config, bool includeAssemblies, out IEnumerable types,
CompressionType compressionType = CompressionType.GZip, CancellationToken cancellationToken = default(CancellationToken)) {
try {
Serializer serializer = new Serializer(obj, config);
if (compressionType == CompressionType.Zip) {
Serialize(stream, includeAssemblies, CompressionLevel.Optimal, serializer, cancellationToken);
} else {
Serialize(stream, serializer, cancellationToken);
}
types = serializer.SerializedTypes;
} catch (PersistenceException) {
throw;
} catch (Exception e) {
throw new PersistenceException("Unexpected exception during Serialization.", e);
}
}
private static void Serialize(Stream stream, Serializer serializer, CancellationToken cancellationToken = default(CancellationToken)) {
cancellationToken.ThrowIfCancellationRequested();
using (StreamWriter writer = new StreamWriter(new GZipStream(stream, CompressionMode.Compress))) {
serializer.InterleaveTypeInformation = true;
XmlGenerator generator = new XmlGenerator();
foreach (ISerializationToken token in serializer) {
cancellationToken.ThrowIfCancellationRequested();
string line = generator.Format(token);
writer.Write(line);
}
writer.Flush();
}
}
}
}