#region License Information
/* HeuristicLab
* Copyright (C) 2002-2012 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.Drawing;
using System.IO;
using HeuristicLab.Clients.Access;
using HeuristicLab.Common;
using HeuristicLab.Core;
using HeuristicLab.Optimization;
using HeuristicLab.Persistence.Default.CompositeSerializers.Storable;
using HeuristicLab.Persistence.Default.Xml;
namespace HeuristicLab.Clients.OKB.RunCreation {
[Item("OKB Run", "The parameters and results of an algorithm run which are stored in the OKB.")]
[StorableClass]
public sealed class OKBRun : NamedItemWrapper, IRun, IStorableContent {
public string Filename { get; set; }
public override Image ItemImage {
get { return Stored ? HeuristicLab.Common.Resources.VSImageLibrary.Database : HeuristicLab.Common.Resources.VSImageLibrary.DatabaseModified; }
}
private long algorithmId;
private long problemId;
private DateTime createdDate;
private bool stored;
public bool Stored {
get { return stored; }
private set {
if (value != stored) {
stored = value;
OnStoredChanged();
OnItemImageChanged();
}
}
}
public IAlgorithm Algorithm {
get { return WrappedItem.Algorithm; }
}
public IDictionary Parameters {
get { return WrappedItem.Parameters; }
}
public IDictionary Results {
get { return WrappedItem.Results; }
}
public IRun WrappedRun {
get { return WrappedItem; }
}
public Color Color {
get { return WrappedItem.Color; }
set { WrappedItem.Color = value; }
}
public bool Visible {
get { return WrappedItem.Visible; }
set { WrappedItem.Visible = value; }
}
#region Persistence Properties
[Storable]
private Guid UserId;
[Storable(Name = "Stored")]
private bool StorableStored {
get { return stored; }
set { stored = value; }
}
[Storable(Name = "AlgorithmId")]
private long StorableAlgorithmId {
get { return algorithmId; }
set { algorithmId = value; }
}
[Storable(Name = "ProblemId")]
private long StorableProblemId {
get { return problemId; }
set { problemId = value; }
}
[Storable(Name = "CreatedDate")]
private DateTime StorableCreatedDate {
get { return createdDate; }
set { createdDate = value; }
}
#endregion
[StorableConstructor]
private OKBRun(bool deserializing) : base(deserializing) { }
private OKBRun(OKBRun original, Cloner cloner)
: base(original, cloner) {
algorithmId = original.algorithmId;
problemId = original.problemId;
createdDate = original.createdDate;
stored = original.stored;
UserId = original.UserId;
}
public OKBRun(long algorithmId, long problemId, IRun run, Guid userId)
: base(run) {
this.algorithmId = algorithmId;
this.problemId = problemId;
this.createdDate = DateTime.Now;
this.stored = false;
this.UserId = userId;
}
public override IDeepCloneable Clone(Cloner cloner) {
return new OKBRun(this, cloner);
}
public void Store() {
if (Stored) throw new InvalidOperationException("Cannot store already stored run.");
Run run = new Run();
run.AlgorithmId = algorithmId;
run.ProblemId = problemId;
//TODO: should there be some error handling? at this point it should already be checked that the user and client are registred
run.UserId = UserId;
run.ClientId = ClientInformation.Instance.ClientInfo.Id;
run.CreatedDate = createdDate;
run.ParameterValues = ConvertToValues(Parameters);
run.ResultValues = ConvertToValues(Results);
RunCreationClient.Instance.AddRun(run);
Stored = true;
}
#region Events
public event EventHandler Changed;
private void OnChanged() {
var handler = Changed;
if (handler != null) handler(this, EventArgs.Empty);
}
public event EventHandler StoredChanged;
private void OnStoredChanged() {
var handler = StoredChanged;
if (handler != null) handler(this, EventArgs.Empty);
}
protected override void RegisterWrappedItemEvents() {
base.RegisterWrappedItemEvents();
WrappedItem.Changed += new EventHandler(WrappedItem_Changed);
}
protected override void DeregisterWrappedItemEvents() {
WrappedItem.Changed -= new EventHandler(WrappedItem_Changed);
base.DeregisterWrappedItemEvents();
}
private void WrappedItem_Changed(object sender, EventArgs e) {
OnChanged();
}
#endregion
#region Helpers
List ConvertToValues(IDictionary items) {
List values = new List();
foreach (var item in items) {
Value value;
if (item.Value is Data.BoolValue) {
BoolValue v = new BoolValue();
v.Value = ((Data.BoolValue)item.Value).Value;
value = v;
} else if (item.Value is Data.IntValue) {
IntValue v = new IntValue();
v.Value = ((Data.IntValue)item.Value).Value;
value = v;
} else if (item.Value is Data.TimeSpanValue) {
TimeSpanValue v = new TimeSpanValue();
v.Value = (long)((Data.TimeSpanValue)item.Value).Value.TotalSeconds;
value = v;
} else if (item.Value is Data.PercentValue) {
PercentValue v = new PercentValue();
v.Value = ((Data.PercentValue)item.Value).Value;
value = v;
} else if (item.Value is Data.DoubleValue) {
DoubleValue v = new DoubleValue();
v.Value = ((Data.DoubleValue)item.Value).Value;
value = v;
} else if (item.Value is Data.StringValue) {
StringValue v = new StringValue();
v.Value = ((Data.StringValue)item.Value).Value;
value = v;
} else {
BinaryValue v = new BinaryValue();
using (MemoryStream stream = new MemoryStream()) {
XmlGenerator.Serialize(item.Value, stream);
stream.Close();
v.Value = stream.ToArray();
}
value = v;
}
value.Name = item.Key;
value.DataType = new DataType();
value.DataType.Name = item.Value.GetType().Name;
value.DataType.TypeName = item.Value.GetType().AssemblyQualifiedName;
values.Add(value);
}
return values;
}
#endregion
}
}