#region License Information
/* HeuristicLab
* Copyright (C) 2002-2019 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.Threading;
using System.Threading.Tasks;
using HeuristicLab.Clients.Access;
using HeuristicLab.Collections;
using HeuristicLab.Common;
using HeuristicLab.Core;
using HeuristicLab.Optimization;
using HEAL.Attic;
using HeuristicLab.Persistence.Default.Xml;
using System;
using System.Collections.Generic;
using System.Drawing;
using System.IO;
using System.Linq;
namespace HeuristicLab.Clients.OKB.RunCreation {
[Item("OKB Algorithm", "An algorithm which is stored in the OKB.")]
[Creatable(CreatableAttribute.Categories.TestingAndAnalysisOKB, Priority = 100)]
[StorableType("159620CA-BC90-48D9-8868-5D8940016BA0")]
public sealed class OKBAlgorithm : Item, IAlgorithm, IStorableContent {
public string Filename { get; set; }
private long algorithmId;
public long AlgorithmId {
get { return algorithmId; }
}
private IAlgorithm algorithm;
private IAlgorithm Algorithm {
get { return algorithm; }
set {
if (value == null) throw new ArgumentNullException("Algorithm", "Algorithm cannot be null.");
if (value != algorithm) {
CancelEventArgs e = new CancelEventArgs(value.Name);
OnNameChanging(e);
if (!e.Cancel) {
IProblem problem = algorithm.Problem;
DeregisterAlgorithmEvents();
algorithm = value;
RegisterAlgorithmEvents();
OnToStringChanged();
OnItemImageChanged();
OnNameChanged();
OnDescriptionChanged();
OnAlgorithmChanged();
OnStoreAlgorithmInEachRunChanged();
try {
algorithm.Problem = problem;
} catch (ArgumentException) {
algorithm.Problem = null;
}
algorithm.Prepare(true);
}
}
}
}
public IEnumerable NestedOptimizers {
get {
// inner algorithm cannot be accessed directly
return Enumerable.Empty();
}
}
public override Image ItemImage {
get { return Algorithm.ItemImage; }
}
public static new Image StaticItemImage {
get { return HeuristicLab.Common.Resources.VSImageLibrary.Event; }
}
public string Name {
get { return Algorithm.Name; }
set { throw new NotSupportedException("Name cannot be changed."); }
}
public string Description {
get { return Algorithm.Description; }
set { throw new NotSupportedException("Description cannot be changed."); }
}
public bool CanChangeName {
get { return false; }
}
public bool CanChangeDescription {
get { return false; }
}
public IKeyedItemCollection Parameters {
get { return Algorithm.Parameters; }
}
public ExecutionState ExecutionState {
get { return Algorithm.ExecutionState; }
}
public TimeSpan ExecutionTime {
get { return Algorithm.ExecutionTime; }
}
public Type ProblemType {
get { return Algorithm.ProblemType; }
}
public IProblem Problem {
get { return Algorithm.Problem; }
set { Algorithm.Problem = value; }
}
public ResultCollection Results {
get { return Algorithm.Results; }
}
private RunCollection runs;
public RunCollection Runs {
get { return runs; }
}
private bool storeRunsAutomatically;
public bool StoreRunsAutomatically {
get { return storeRunsAutomatically; }
set {
if (value != storeRunsAutomatically) {
storeRunsAutomatically = value;
OnStoreRunsAutomaticallyChanged();
}
}
}
public bool StoreAlgorithmInEachRun {
get { return Algorithm.StoreAlgorithmInEachRun; }
set { Algorithm.StoreAlgorithmInEachRun = value; }
}
#region Persistence Properties
[Storable]
private Guid UserId;
[Storable(Name = "AlgorithmId")]
private long StorableAlgorithmId {
get { return algorithmId; }
set { algorithmId = value; }
}
[Storable(Name = "Algorithm")]
private IAlgorithm StorableAlgorithm {
get { return algorithm; }
set {
algorithm = value;
RegisterAlgorithmEvents();
}
}
[Storable(Name = "Runs")]
private RunCollection StorableRuns {
get { return runs; }
set {
runs = value;
RegisterRunsEvents();
}
}
[Storable(Name = "StoreRunsAutomatically")]
private bool StorableStoreRunsAutomatically {
get { return storeRunsAutomatically; }
set { storeRunsAutomatically = value; }
}
#endregion
[StorableConstructor]
private OKBAlgorithm(StorableConstructorFlag _) : base(_) { }
private OKBAlgorithm(OKBAlgorithm original, Cloner cloner)
: base(original, cloner) {
algorithmId = original.algorithmId;
algorithm = cloner.Clone(original.algorithm);
RegisterAlgorithmEvents();
runs = cloner.Clone(original.runs);
storeRunsAutomatically = original.storeRunsAutomatically;
UserId = original.UserId;
RegisterRunsEvents();
}
public OKBAlgorithm()
: base() {
algorithmId = -1;
algorithm = new EmptyAlgorithm("No algorithm selected. Please choose an algorithm from the OKB.");
RegisterAlgorithmEvents();
runs = new RunCollection();
storeRunsAutomatically = true;
RegisterRunsEvents();
}
private void CheckUserPermissions() {
if (UserInformation.Instance.UserExists) {
if (UserInformation.Instance.User.Roles.Count(x => x.Name == OKBRoles.OKBUser || x.Name == OKBRoles.OKBAdministrator) > 0) {
UserId = UserInformation.Instance.User.Id;
} else {
throw new Exception("You don't have the appropriate roles for executing OKB Algorithms.");
}
} else {
throw new Exception("You need an user account for executing OKB Algorithms.");
}
}
public override IDeepCloneable Clone(Cloner cloner) {
return new OKBAlgorithm(this, cloner);
}
public void Load(long algorithmId) {
if (this.algorithmId != algorithmId) {
IAlgorithm algorithm;
byte[] algorithmData = RunCreationClient.Instance.GetAlgorithmData(algorithmId);
using (MemoryStream stream = new MemoryStream(algorithmData)) {
algorithm = XmlParser.Deserialize(stream);
}
this.algorithmId = algorithmId;
Algorithm = algorithm;
}
}
public IAlgorithm CloneAlgorithm() {
return (IAlgorithm)Algorithm.Clone();
}
public void CollectParameterValues(IDictionary values) {
Algorithm.CollectParameterValues(values);
}
public void CollectResultValues(IDictionary values) {
Algorithm.CollectResultValues(values);
}
public void Prepare() {
Algorithm.Prepare();
}
public void Prepare(bool clearRuns) {
if (clearRuns) runs.Clear();
Algorithm.Prepare(clearRuns);
}
public void Start() {
Start(CancellationToken.None);
}
public void Start(CancellationToken cancellationToken) {
CheckUserPermissions();
if (!ClientInformation.Instance.ClientExists && storeRunsAutomatically) {
throw new MissingClientRegistrationException();
}
Algorithm.Start(cancellationToken);
}
public async Task StartAsync() { await StartAsync(CancellationToken.None); }
public async Task StartAsync(CancellationToken cancellationToken) {
await AsyncHelper.DoAsync(Start, cancellationToken);
}
public void Pause() {
Algorithm.Pause();
}
public void Stop() {
Algorithm.Stop();
}
#region Events
public event EventHandler AlgorithmChanged;
private void OnAlgorithmChanged() {
EventHandler handler = AlgorithmChanged;
if (handler != null) handler(this, EventArgs.Empty);
}
public event EventHandler> NameChanging;
private void OnNameChanging(CancelEventArgs e) {
var handler = NameChanging;
if (handler != null) handler(this, e);
}
public event EventHandler NameChanged;
private void OnNameChanged() {
var handler = NameChanged;
if (handler != null) handler(this, EventArgs.Empty);
}
public event EventHandler DescriptionChanged;
private void OnDescriptionChanged() {
var handler = DescriptionChanged;
if (handler != null) handler(this, EventArgs.Empty);
}
public event EventHandler ExecutionStateChanged;
private void OnExecutionStateChanged() {
var handler = ExecutionStateChanged;
if (handler != null) handler(this, EventArgs.Empty);
}
public event EventHandler ExecutionTimeChanged;
private void OnExecutionTimeChanged() {
var handler = ExecutionTimeChanged;
if (handler != null) handler(this, EventArgs.Empty);
}
public event EventHandler ProblemChanged;
private void OnProblemChanged() {
var handler = ProblemChanged;
if (handler != null) handler(this, EventArgs.Empty);
}
public event EventHandler StoreRunsAutomaticallyChanged;
private void OnStoreRunsAutomaticallyChanged() {
var handler = StoreRunsAutomaticallyChanged;
if (handler != null) handler(this, EventArgs.Empty);
}
public event EventHandler StoreAlgorithmInEachRunChanged;
private void OnStoreAlgorithmInEachRunChanged() {
var handler = StoreAlgorithmInEachRunChanged;
if (handler != null) handler(this, EventArgs.Empty);
}
public event EventHandler Prepared;
private void OnPrepared() {
var handler = Prepared;
if (handler != null) handler(this, EventArgs.Empty);
}
public event EventHandler Started;
private void OnStarted() {
var handler = Started;
if (handler != null) handler(this, EventArgs.Empty);
}
public event EventHandler Paused;
private void OnPaused() {
var handler = Paused;
if (handler != null) handler(this, EventArgs.Empty);
}
public event EventHandler Stopped;
private void OnStopped() {
var handler = Stopped;
if (handler != null) handler(this, EventArgs.Empty);
}
public event EventHandler> ExceptionOccurred;
private void OnExceptionOccurred(Exception exception) {
var handler = ExceptionOccurred;
if (handler != null) handler(this, new EventArgs(exception));
}
private void RegisterAlgorithmEvents() {
if (Algorithm != null) {
Algorithm.ToStringChanged += new EventHandler(Algorithm_ToStringChanged);
Algorithm.ItemImageChanged += new EventHandler(Algorithm_ItemImageChanged);
Algorithm.NameChanging += new EventHandler>(Algorithm_NameChanging);
Algorithm.NameChanged += new EventHandler(Algorithm_NameChanged);
Algorithm.DescriptionChanged += new EventHandler(Algorithm_DescriptionChanged);
Algorithm.ExecutionStateChanged += new EventHandler(Algorithm_ExecutionStateChanged);
Algorithm.ExecutionTimeChanged += new EventHandler(Algorithm_ExecutionTimeChanged);
Algorithm.ProblemChanged += new EventHandler(Algorithm_ProblemChanged);
Algorithm.Runs.ItemsAdded += new CollectionItemsChangedEventHandler(Algorithm_Runs_ItemsAdded);
Algorithm.StoreAlgorithmInEachRunChanged += new EventHandler(Algorithm_StoreAlgorithmInEachRunChanged);
Algorithm.Prepared += new EventHandler(Algorithm_Prepared);
Algorithm.Started += new EventHandler(Algorithm_Started);
Algorithm.Paused += new EventHandler(Algorithm_Paused);
Algorithm.Stopped += new EventHandler(Algorithm_Stopped);
Algorithm.ExceptionOccurred += new EventHandler>(Algorithm_ExceptionOccurred);
}
}
private void DeregisterAlgorithmEvents() {
if (Algorithm != null) {
Algorithm.ToStringChanged -= new EventHandler(Algorithm_ToStringChanged);
Algorithm.ItemImageChanged -= new EventHandler(Algorithm_ItemImageChanged);
Algorithm.NameChanging -= new EventHandler>(Algorithm_NameChanging);
Algorithm.NameChanged -= new EventHandler(Algorithm_NameChanged);
Algorithm.DescriptionChanged -= new EventHandler(Algorithm_DescriptionChanged);
Algorithm.ExecutionStateChanged -= new EventHandler(Algorithm_ExecutionStateChanged);
Algorithm.ExecutionTimeChanged -= new EventHandler(Algorithm_ExecutionTimeChanged);
Algorithm.ProblemChanged -= new EventHandler(Algorithm_ProblemChanged);
Algorithm.Runs.ItemsAdded -= new CollectionItemsChangedEventHandler(Algorithm_Runs_ItemsAdded);
Algorithm.StoreAlgorithmInEachRunChanged -= new EventHandler(Algorithm_StoreAlgorithmInEachRunChanged);
Algorithm.Prepared -= new EventHandler(Algorithm_Prepared);
Algorithm.Started -= new EventHandler(Algorithm_Started);
Algorithm.Paused -= new EventHandler(Algorithm_Paused);
Algorithm.Stopped -= new EventHandler(Algorithm_Stopped);
Algorithm.ExceptionOccurred -= new EventHandler>(Algorithm_ExceptionOccurred);
}
}
private void Algorithm_ToStringChanged(object sender, EventArgs e) {
OnToStringChanged();
}
private void Algorithm_ItemImageChanged(object sender, EventArgs e) {
OnItemImageChanged();
}
private void Algorithm_NameChanging(object sender, CancelEventArgs e) {
OnNameChanging(e);
}
private void Algorithm_NameChanged(object sender, EventArgs e) {
OnNameChanged();
}
private void Algorithm_DescriptionChanged(object sender, EventArgs e) {
OnDescriptionChanged();
}
private void Algorithm_ExecutionStateChanged(object sender, EventArgs e) {
OnExecutionStateChanged();
}
private void Algorithm_ExecutionTimeChanged(object sender, EventArgs e) {
OnExecutionTimeChanged();
}
private void Algorithm_ProblemChanged(object sender, EventArgs e) {
OnProblemChanged();
}
private void Algorithm_Runs_ItemsAdded(object sender, CollectionItemsChangedEventArgs e) {
OKBProblem problem = Problem as OKBProblem;
foreach (IRun run in e.Items) {
if (problem != null) {
OKBRun okbRun = new OKBRun(AlgorithmId, problem.ProblemId, run, UserId);
runs.Add(okbRun);
if (StoreRunsAutomatically) {
okbRun.Store();
}
} else {
runs.Add(run);
}
}
}
private void Algorithm_StoreAlgorithmInEachRunChanged(object sender, EventArgs e) {
OnStoreAlgorithmInEachRunChanged();
}
private void Algorithm_Prepared(object sender, EventArgs e) {
OnPrepared();
}
private void Algorithm_Started(object sender, EventArgs e) {
OnStarted();
}
private void Algorithm_Paused(object sender, EventArgs e) {
OnPaused();
}
private void Algorithm_Stopped(object sender, EventArgs e) {
OnStopped();
}
private void Algorithm_ExceptionOccurred(object sender, EventArgs e) {
OnExceptionOccurred(e.Value);
}
private void RegisterRunsEvents() {
runs.ItemsRemoved += new CollectionItemsChangedEventHandler(runs_ItemsRemoved);
runs.CollectionReset += new CollectionItemsChangedEventHandler(Runs_CollectionReset);
}
private void runs_ItemsRemoved(object sender, CollectionItemsChangedEventArgs e) {
foreach (IRun run in e.Items) {
OKBRun okbRun = run as OKBRun;
if (okbRun != null)
Algorithm.Runs.Remove(okbRun.WrappedRun);
else
Algorithm.Runs.Remove(run);
}
}
private void Runs_CollectionReset(object sender, CollectionItemsChangedEventArgs e) {
foreach (IRun run in e.OldItems) {
OKBRun okbRun = run as OKBRun;
if (okbRun != null)
Algorithm.Runs.Remove(okbRun.WrappedRun);
else
Algorithm.Runs.Remove(run);
}
}
#endregion
}
}