#region License Information /* HeuristicLab * Copyright (C) 2002-2015 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.Linq; using System.ServiceModel.Security; using System.Threading; using System.Threading.Tasks; using System.Windows.Forms; using HeuristicLab.Clients.Access; using HeuristicLab.Clients.Hive.Views; using HeuristicLab.Core; using HeuristicLab.Core.Views; using HeuristicLab.MainForm; using TS = System.Threading.Tasks; namespace HeuristicLab.Clients.Hive.Administrator.Views { [View("Resources View")] [Content(typeof(IItemList), false)] public partial class ResourcesView : ItemView, IDisposable { public new IItemList Content { get { return (IItemList)base.Content; } set { base.Content = value; } } public const string UngroupedGroupName = "UNGROUPED"; private const int slaveImageIndex = 0; private const int slaveGroupImageIndex = 1; private readonly Color ownedResourceColor = Color.LightGreen; private TS.Task progressTask; private bool stopProgressTask; private bool currentlyAuthorized; public ResourcesView() { InitializeComponent(); treeSlaveGroup.ImageList.Images.Add(HeuristicLab.Common.Resources.VSImageLibrary.MonitorLarge); treeSlaveGroup.ImageList.Images.Add(HeuristicLab.Common.Resources.VSImageLibrary.NetworkCenterLarge); HiveAdminClient.Instance.Refreshing += new EventHandler(Instance_Refreshing); HiveAdminClient.Instance.Refreshed += new EventHandler(Instance_Refreshed); Access.AccessClient.Instance.Refreshing += new EventHandler(AccessClient_Refreshing); Access.AccessClient.Instance.Refreshed += new EventHandler(AccessClient_Refreshed); } private void UpdateProgress() { while (!stopProgressTask) { int diff = (progressBar.Maximum - progressBar.Minimum) / 10; if (progressBar.InvokeRequired) { progressBar.Invoke(new Action(delegate() { progressBar.Value = (progressBar.Value + diff) % progressBar.Maximum; })); } else { progressBar.Value = (progressBar.Value + diff) % progressBar.Maximum; } //ok, this is not very clever... Thread.Sleep(500); } if (progressBar.InvokeRequired) { progressBar.Invoke(new Action(delegate() { progressBar.Value = progressBar.Minimum; })); } else { progressBar.Value = progressBar.Minimum; } } void Instance_Refreshing(object sender, EventArgs e) { stopProgressTask = false; progressTask = new TS.Task(UpdateProgress); progressTask.Start(); SetEnabledStateOfControls(); } void Instance_Refreshed(object sender, EventArgs e) { stopProgressTask = true; SetEnabledStateOfControls(); } void AccessClient_Refreshing(object sender, EventArgs e) { stopProgressTask = false; progressTask = new TS.Task(UpdateProgress); progressTask.Start(); SetEnabledStateOfControls(); btnPermissionsSave.Enabled = false; } void AccessClient_Refreshed(object sender, EventArgs e) { stopProgressTask = true; SetEnabledStateOfControls(); } #region Register Content Events protected override void DeregisterContentEvents() { Content.ItemsAdded -= new Collections.CollectionItemsChangedEventHandler>(Content_ItemsAdded); Content.ItemsRemoved -= new Collections.CollectionItemsChangedEventHandler>(Content_ItemsRemoved); base.DeregisterContentEvents(); } protected override void RegisterContentEvents() { base.RegisterContentEvents(); Content.ItemsAdded += new Collections.CollectionItemsChangedEventHandler>(Content_ItemsAdded); Content.ItemsRemoved += new Collections.CollectionItemsChangedEventHandler>(Content_ItemsRemoved); } #endregion protected override void OnContentChanged() { base.OnContentChanged(); if (Content == null) { slaveView.Content = null; scheduleView.Content = null; permissionView.Content = null; permissionView.FetchSelectedUsers = null; treeSlaveGroup.Nodes.Clear(); } else { permissionView.Content = Access.AccessClient.Instance; treeSlaveGroup.Nodes.Clear(); //rebuild TreeNode ungrp = new TreeNode(UngroupedGroupName); ungrp.ImageIndex = slaveGroupImageIndex; ungrp.SelectedImageIndex = ungrp.ImageIndex; var newGroup = new SlaveGroup(); newGroup.Name = UngroupedGroupName; newGroup.Id = Guid.NewGuid(); newGroup.Description = "Contains slaves which are in no group"; ungrp.Tag = newGroup; foreach (Resource g in Content) { if (g.GetType() == typeof(SlaveGroup)) { //root node if (g.ParentResourceId == null) { TreeNode tn = new TreeNode(); tn.ImageIndex = slaveGroupImageIndex; tn.SelectedImageIndex = tn.ImageIndex; tn.Tag = g; tn.Text = g.Name; if (g.OwnerUserId == Access.UserInformation.Instance.User.Id) tn.BackColor = ownedResourceColor; BuildSlaveGroupTree(g, tn); treeSlaveGroup.Nodes.Add(tn); } } else if (g.GetType() == typeof(Slave)) { if (g.ParentResourceId == null) { var stn = new TreeNode(g.Name); stn.ImageIndex = slaveImageIndex; stn.SelectedImageIndex = stn.ImageIndex; stn.Tag = g; if (g.OwnerUserId == Access.UserInformation.Instance.User.Id) stn.BackColor = ownedResourceColor; ungrp.Nodes.Add(stn); } } } treeSlaveGroup.Nodes.Add(ungrp); } } private void BuildSlaveGroupTree(Resource g, TreeNode tn) { foreach (Resource r in Content.Where(s => s.ParentResourceId != null && s.ParentResourceId == g.Id)) { TreeNode stn = new TreeNode(r.Name); if (r is Slave) { stn.ImageIndex = slaveImageIndex; } else if (r is SlaveGroup) { stn.ImageIndex = slaveGroupImageIndex; } stn.SelectedImageIndex = stn.ImageIndex; stn.Tag = r; if (r.OwnerUserId == Access.UserInformation.Instance.User.Id) stn.BackColor = ownedResourceColor; tn.Nodes.Add(stn); BuildSlaveGroupTree(r, stn); } } protected override void SetEnabledStateOfControls() { base.SetEnabledStateOfControls(); if (Content == null) { btnAddGroup.Enabled = false; btnRemoveGroup.Enabled = false; btnSave.Enabled = false; btnPermissionsSave.Enabled = false; permissionView.Enabled = false; scheduleView.SetEnabledStateOfSchedule(false); btnPermissionsSave.Enabled = false; permissionView.Enabled = false; } else { btnAddGroup.Enabled = true; btnRemoveGroup.Enabled = true; btnSave.Enabled = true; scheduleView.SetEnabledStateOfSchedule(IsAuthorized(slaveView.Content)); btnPermissionsSave.Enabled = permissionView.FetchSelectedUsers != null; permissionView.Enabled = permissionView.FetchSelectedUsers != null; } } private bool IsAuthorized(Resource resource) { return resource != null && resource.Name != UngroupedGroupName && resource.Id != Guid.Empty && UserInformation.Instance.UserExists && (resource.OwnerUserId == UserInformation.Instance.User.Id || HiveRoles.CheckAdminUserPermissions()); } private void treeSlaveGroup_AfterSelect(object sender, TreeViewEventArgs e) { if (e.Action != TreeViewAction.Unknown) { Resource selectedResource = ((Resource)e.Node.Tag); currentlyAuthorized = IsAuthorized(selectedResource); if (currentlyAuthorized) { permissionView.FetchSelectedUsers = new Func>(() => { return HiveServiceLocator.Instance.CallHiveService>(service => { return service.GetResourcePermissions(selectedResource.Id); }).Select(x => x.GrantedUserId).ToList(); }); if (!tabSlaveGroup.TabPages.Contains(tabPermissions)) tabSlaveGroup.TabPages.Add(tabPermissions); } else { permissionView.FetchSelectedUsers = null; btnPermissionsSave.Enabled = false; if (selectedResource.Id == Guid.Empty) { if (!tabSlaveGroup.TabPages.Contains(tabPermissions)) tabSlaveGroup.TabPages.Add(tabPermissions); } else tabSlaveGroup.TabPages.Remove(tabPermissions); } if (slaveView.Content != null && slaveView.Content is SlaveGroup) { slaveView.Content.PropertyChanged -= new System.ComponentModel.PropertyChangedEventHandler(SlaveViewContent_PropertyChanged); } slaveView.Content = selectedResource; HiveAdminClient.Instance.DowntimeForResourceId = selectedResource.Id; if (selectedResource is SlaveGroup) { slaveView.Content.PropertyChanged += new System.ComponentModel.PropertyChangedEventHandler(SlaveViewContent_PropertyChanged); } if (tabSlaveGroup.SelectedIndex == 1) { UpdateScheduleAsync(); } else if (tabSlaveGroup.SelectedIndex == 2) { UpdatePermissionsAsync(); } } } void SlaveViewContent_PropertyChanged(object sender, System.ComponentModel.PropertyChangedEventArgs e) { OnContentChanged(); if (e.PropertyName == "HbInterval") { UpdateChildHbIntervall(slaveView.Content); } } private void UpdateChildHbIntervall(Resource resource) { foreach (Resource r in Content.Where(x => x.ParentResourceId == resource.Id)) { r.HbInterval = resource.HbInterval; if (r is SlaveGroup) { UpdateChildHbIntervall(r); } } } private void btnAddGroup_Click(object sender, EventArgs e) { SlaveGroup newGroup = new SlaveGroup(); newGroup.Name = "New Group"; newGroup.OwnerUserId = UserInformation.Instance.User.Id; Content.Add(newGroup); } void Content_ItemsRemoved(object sender, Collections.CollectionItemsChangedEventArgs> e) { OnContentChanged(); } void Content_ItemsAdded(object sender, Collections.CollectionItemsChangedEventArgs> e) { OnContentChanged(); } private void btnRemoveGroup_Click(object sender, EventArgs e) { if (treeSlaveGroup.SelectedNode != null && treeSlaveGroup.SelectedNode.Tag != null) { Resource res = (Resource)treeSlaveGroup.SelectedNode.Tag; DialogResult diagRes = MessageBox.Show("Do you really want to delete " + res.Name + "?", "HeuristicLab Hive Administrator", MessageBoxButtons.YesNo, MessageBoxIcon.Question); if (diagRes == DialogResult.Yes) { if (res is Slave) { Content.Remove(res); HiveAdminClient.Delete(res); } else if (res is SlaveGroup) { //only delete empty groups if (Content.Where(s => s.ParentResourceId == res.Id).Count() < 1) { Content.Remove(res); HiveAdminClient.Delete(res); } else { MessageBox.Show("Only empty groups can be deleted.", "HeuristicLab Hive Administrator", MessageBoxButtons.OK, MessageBoxIcon.Error); } } } } } private void btnSave_Click(object sender, EventArgs e) { foreach (Resource res in Content) { if (res is SlaveGroup && res.Id == Guid.Empty) { SlaveGroup slaveGroup = (SlaveGroup)res; slaveGroup.Store(); } else if (res.Id != Guid.Empty && res.Modified) { res.Store(); } } } private void treeSlaveGroup_DragDrop(object sender, DragEventArgs e) { if (e.Data.GetDataPresent("System.Windows.Forms.TreeNode", false)) { Point pt = ((TreeView)sender).PointToClient(new Point(e.X, e.Y)); TreeNode destNode = ((TreeView)sender).GetNodeAt(pt); TreeNode newNode = (TreeNode)e.Data.GetData("System.Windows.Forms.TreeNode"); if (destNode.TreeView == newNode.TreeView) { if (destNode.Text == UngroupedGroupName || (destNode.Parent != null && destNode.Parent.Text == UngroupedGroupName)) { MessageBox.Show(string.Format("You can't drag items to the group \"{0}\".{1}This group only contains slaves which haven't yet been assigned to a real group.", UngroupedGroupName, Environment.NewLine), "HeuristicLab Hive Administrator", MessageBoxButtons.OK, MessageBoxIcon.Information); return; } SlaveGroup sgrp = null; TreeNode parentNode = null; if (destNode.Tag != null && destNode.Tag is SlaveGroup) { sgrp = (SlaveGroup)destNode.Tag; parentNode = destNode; } else if (destNode.Parent != null && destNode.Parent.Tag is SlaveGroup) { sgrp = (SlaveGroup)destNode.Parent.Tag; parentNode = destNode.Parent; } if (newNode.Tag is SlaveGroup && CheckParentsEqualsMovedNode(parentNode, newNode)) { return; } SlaveGroup parent = (SlaveGroup)parentNode.Tag; if (parent.OwnerUserId != null && !IsAuthorized(parent)) { MessageBox.Show(string.Format("You don't have the permissions to drag items to the group \"{0}\".", ((Resource)parentNode.Tag).Name), "HeuristicLab Hive Administrator", MessageBoxButtons.OK, MessageBoxIcon.Error); return; } if (sgrp != null && newNode.Tag != null) { //save parent group to get an id if (sgrp.Id == Guid.Empty) { sgrp.Store(); } if (newNode.Tag is Slave) { Slave slave = (Slave)newNode.Tag; if (slave.ParentResourceId == null || (slave.ParentResourceId != null && slave.ParentResourceId != sgrp.Id)) { slave.ParentResourceId = sgrp.Id; newNode.Remove(); parentNode.Nodes.Clear(); BuildSlaveGroupTree(sgrp, parentNode); } } else if (newNode.Tag is SlaveGroup) { SlaveGroup slaveGroup = (SlaveGroup)newNode.Tag; if (slaveGroup.ParentResourceId == null || (slaveGroup.ParentResourceId != null && slaveGroup.ParentResourceId != sgrp.Id)) { slaveGroup.ParentResourceId = sgrp.Id; newNode.Remove(); parentNode.Nodes.Clear(); BuildSlaveGroupTree(sgrp, parentNode); } } } } } } private bool CheckParentsEqualsMovedNode(TreeNode dest, TreeNode movedNode) { TreeNode tmp = dest; while (tmp != null) { if (tmp == movedNode) { return true; } tmp = tmp.Parent; } return false; } private void treeSlaveGroup_ItemDrag(object sender, ItemDragEventArgs e) { TreeNode sourceNode = (TreeNode)e.Item; if (IsAuthorized((Resource)sourceNode.Tag)) DoDragDrop(sourceNode, DragDropEffects.All); } private void treeSlaveGroup_DragEnter(object sender, DragEventArgs e) { e.Effect = DragDropEffects.Move; } private void treeSlaveGroup_DragOver(object sender, DragEventArgs e) { e.Effect = DragDropEffects.Move; } private void treeSlaveGroup_QueryContinueDrag(object sender, QueryContinueDragEventArgs e) { e.Action = DragAction.Continue; } void ResetView() { if (this.InvokeRequired) { Invoke(new Action(ResetView)); } else { treeSlaveGroup.Nodes.Clear(); if (slaveView.Content != null && slaveView.Content is SlaveGroup) { slaveView.Content.PropertyChanged -= new System.ComponentModel.PropertyChangedEventHandler(SlaveViewContent_PropertyChanged); } slaveView.Content = null; if (scheduleView.Content != null) { scheduleView.Content.Clear(); } HiveAdminClient.Instance.ResetDowntime(); } } private void UpdateResources() { ResetView(); try { if (!Access.UserInformation.Instance.UserExists) { //do a refresh just in case that the user has changed his usr and pwd in between Access.UserInformation.Instance.Refresh(); } HiveAdminClient.Instance.Refresh(); Content = HiveAdminClient.Instance.Resources; } catch (MessageSecurityException) { ShowMessageSecurityException(); } catch (AnonymousUserException) { ShowHiveInformationDialog(); } } private void ShowMessageSecurityException() { if (this.InvokeRequired) { Invoke(new Action(ShowMessageSecurityException)); } else { MessageBox.Show("A Message Security error has occured. This normally means that your user name or password is wrong.", "HeuristicLab Hive Administrator", MessageBoxButtons.OK, MessageBoxIcon.Error); } } private void ShowHiveInformationDialog() { if (this.InvokeRequired) { Invoke(new Action(ShowHiveInformationDialog)); } else { using (HiveInformationDialog dialog = new HiveInformationDialog()) { dialog.ShowDialog(this); } } } private void UpdateResourcesAsync() { TS.Task.Factory.StartNew(UpdateResources).ContinueWith((t) => { DisplayError(t.Exception); }, TaskContinuationOptions.OnlyOnFaulted); } private void UpdateSchedule() { HiveAdminClient.Instance.RefreshCalendar(); scheduleView.Invoke(new Action(() => { scheduleView.Content = HiveAdminClient.Instance.Downtimes; SetEnabledStateOfControls(); })); } private void UpdateScheduleAsync() { TS.Task.Factory.StartNew(UpdateSchedule).ContinueWith((t) => { DisplayError(t.Exception); }, TaskContinuationOptions.OnlyOnFaulted); } private void UpdatePermissions() { if (permissionView.Content != null && permissionView.FetchSelectedUsers != null) permissionView.Invoke(new Action(() => permissionView.ManualRefresh())); } private void UpdatePermissionsAsync() { TS.Task.Factory.StartNew(UpdatePermissions).ContinueWith((t) => { DisplayError(t.Exception); }, TaskContinuationOptions.OnlyOnFaulted); } private void DisplayError(Exception ex) { MessageBox.Show(string.Format("An error occured while updating: {0} {1}", Environment.NewLine, ex.Message), "HeuristicLab Hive Administrator", MessageBoxButtons.OK, MessageBoxIcon.Error); } private void tabSlaveGroup_SelectedIndexChanged(object sender, EventArgs e) { if (tabSlaveGroup.SelectedIndex == 1) { UpdateScheduleAsync(); } else if (tabSlaveGroup.SelectedIndex == 2) { UpdatePermissionsAsync(); } } private void btnRefresh_Click(object sender, EventArgs e) { UpdateResourcesAsync(); } private void ResourcesView_Load(object sender, EventArgs e) { UpdateResourcesAsync(); } private void btnPermissionsSave_Click(object sender, EventArgs e) { SetEnabledStateOfControls(); HiveServiceLocator.Instance.CallHiveService(service => { service.GrantResourcePermissions(((Resource)treeSlaveGroup.SelectedNode.Tag).Id, permissionView.GetAddedUsers().Select(x => x.Id).ToList()); service.RevokeResourcePermissions(((Resource)treeSlaveGroup.SelectedNode.Tag).Id, permissionView.GetDeletedUsers().Select(x => x.Id).ToList()); }); SetEnabledStateOfControls(); } } }