[6976] | 1 | #region License Information
|
---|
| 2 | /* HeuristicLab
|
---|
[16117] | 3 | * Copyright (C) 2002-2017 Heuristic and Evolutionary Algorithms Laboratory (HEAL)
|
---|
[6976] | 4 | *
|
---|
| 5 | * This file is part of HeuristicLab.
|
---|
| 6 | *
|
---|
| 7 | * HeuristicLab is free software: you can redistribute it and/or modify
|
---|
| 8 | * it under the terms of the GNU General Public License as published by
|
---|
| 9 | * the Free Software Foundation, either version 3 of the License, or
|
---|
| 10 | * (at your option) any later version.
|
---|
| 11 | *
|
---|
| 12 | * HeuristicLab is distributed in the hope that it will be useful,
|
---|
| 13 | * but WITHOUT ANY WARRANTY; without even the implied warranty of
|
---|
| 14 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
---|
| 15 | * GNU General Public License for more details.
|
---|
| 16 | *
|
---|
| 17 | * You should have received a copy of the GNU General Public License
|
---|
| 18 | * along with HeuristicLab. If not, see <http://www.gnu.org/licenses/>.
|
---|
| 19 | */
|
---|
| 20 | #endregion
|
---|
| 21 |
|
---|
| 22 | using System;
|
---|
[8051] | 23 | using System.Collections.Generic;
|
---|
[16117] | 24 | using System.ComponentModel;
|
---|
[6976] | 25 | using System.Drawing;
|
---|
| 26 | using System.Linq;
|
---|
| 27 | using System.Windows.Forms;
|
---|
[8051] | 28 | using HeuristicLab.Clients.Access;
|
---|
[6976] | 29 | using HeuristicLab.Clients.Hive.Views;
|
---|
[16117] | 30 | using HeuristicLab.Collections;
|
---|
| 31 | using HeuristicLab.Common.Resources;
|
---|
[6976] | 32 | using HeuristicLab.Core;
|
---|
| 33 | using HeuristicLab.Core.Views;
|
---|
| 34 | using HeuristicLab.MainForm;
|
---|
| 35 |
|
---|
| 36 | namespace HeuristicLab.Clients.Hive.Administrator.Views {
|
---|
| 37 | [View("Resources View")]
|
---|
[7928] | 38 | [Content(typeof(IItemList<Resource>), false)]
|
---|
[6976] | 39 | public partial class ResourcesView : ItemView, IDisposable {
|
---|
| 40 | private const int slaveImageIndex = 0;
|
---|
| 41 | private const int slaveGroupImageIndex = 1;
|
---|
[16117] | 42 | public const string UNGROUPED_GROUP_NAME = "UNGROUPED";
|
---|
| 43 | public const string UNGROUPED_GROUP_DESCRIPTION = "Contains slaves that are not assigned to any group.";
|
---|
| 44 | private const string SELECTED_TAG = ""; // " [selected]";
|
---|
| 45 | private const string NOT_STORED_TAG = "*"; // " [not stored]";
|
---|
| 46 | private const string CHANGES_NOT_STORED_TAG = "*"; // " [changes not stored]";
|
---|
[6976] | 47 |
|
---|
[16117] | 48 | private readonly Color changedColor = Color.FromArgb(255, 87, 191, 193); // #57bfc1
|
---|
| 49 | private readonly Color selectedBackColor = Color.DodgerBlue;
|
---|
| 50 | private readonly Color selectedForeColor = Color.White;
|
---|
| 51 | private readonly Color calculatingColor = Color.FromArgb(255, 58, 114, 35); // #3a7223
|
---|
| 52 | private readonly Color offlineColor = Color.FromArgb(255, 187, 36, 36); // #bb2424
|
---|
| 53 | private readonly Color grayTextColor = SystemColors.GrayText;
|
---|
[6976] | 54 |
|
---|
| 55 |
|
---|
[8051] | 56 |
|
---|
[16117] | 57 | private TreeNode ungroupedGroupNode;
|
---|
| 58 |
|
---|
| 59 | private Resource selectedResource = null;
|
---|
| 60 | public Resource SelectedResource {
|
---|
| 61 | get { return selectedResource; }
|
---|
| 62 | set { if (selectedResource != value) ChangeSelectedResource(value); }
|
---|
[6976] | 63 | }
|
---|
| 64 |
|
---|
[16117] | 65 | private readonly object locker = new object();
|
---|
[6976] | 66 |
|
---|
[16117] | 67 | public new IItemList<Resource> Content {
|
---|
| 68 | get { return (IItemList<Resource>)base.Content; }
|
---|
| 69 | set { base.Content = value; }
|
---|
[6976] | 70 | }
|
---|
| 71 |
|
---|
[16117] | 72 | public ResourcesView() {
|
---|
| 73 | InitializeComponent();
|
---|
[6976] | 74 |
|
---|
[16117] | 75 | treeView.ImageList.Images.Add(VSImageLibrary.MonitorLarge);
|
---|
| 76 | treeView.ImageList.Images.Add(VSImageLibrary.NetworkCenterLarge);
|
---|
| 77 |
|
---|
| 78 | HiveAdminClient.Instance.Refreshing += HiveAdminClient_Instance_Refreshing;
|
---|
| 79 | HiveAdminClient.Instance.Refreshed += HiveAdminClient_Instance_Refreshed;
|
---|
| 80 | AccessClient.Instance.Refreshing += AccessClient_Instance_Refreshing;
|
---|
| 81 | AccessClient.Instance.Refreshed += AccessClient_Instance_Refreshed;
|
---|
[6976] | 82 | }
|
---|
| 83 |
|
---|
[16117] | 84 | #region Overrides
|
---|
| 85 | protected override void OnClosing(FormClosingEventArgs e) {
|
---|
| 86 | AccessClient.Instance.Refreshed -= AccessClient_Instance_Refreshed;
|
---|
| 87 | AccessClient.Instance.Refreshing -= AccessClient_Instance_Refreshing;
|
---|
| 88 | HiveAdminClient.Instance.Refreshed -= HiveAdminClient_Instance_Refreshed;
|
---|
| 89 | HiveAdminClient.Instance.Refreshing -= HiveAdminClient_Instance_Refreshing;
|
---|
| 90 | base.OnClosing(e);
|
---|
[8051] | 91 | }
|
---|
| 92 |
|
---|
[16117] | 93 | protected override void RegisterContentEvents() {
|
---|
| 94 | base.RegisterContentEvents();
|
---|
| 95 | Content.ItemsAdded += Content_ItemsAdded;
|
---|
| 96 | Content.ItemsRemoved += Content_ItemsRemoved;
|
---|
[8051] | 97 | }
|
---|
| 98 |
|
---|
[6976] | 99 | protected override void DeregisterContentEvents() {
|
---|
[16117] | 100 | Content.ItemsRemoved -= Content_ItemsRemoved;
|
---|
| 101 | Content.ItemsAdded -= Content_ItemsAdded;
|
---|
[6976] | 102 | base.DeregisterContentEvents();
|
---|
| 103 | }
|
---|
| 104 |
|
---|
| 105 | protected override void OnContentChanged() {
|
---|
| 106 | base.OnContentChanged();
|
---|
| 107 | if (Content == null) {
|
---|
[16117] | 108 | treeView.Nodes.Clear();
|
---|
| 109 | viewHost.Content = null;
|
---|
[8051] | 110 | scheduleView.Content = null;
|
---|
[6976] | 111 | } else {
|
---|
[16117] | 112 | BuildResourceTree(Content);
|
---|
| 113 | }
|
---|
[16122] | 114 | SetEnabledStateOfControls();
|
---|
[16117] | 115 | }
|
---|
[6976] | 116 |
|
---|
[16117] | 117 | protected override void SetEnabledStateOfControls() {
|
---|
| 118 | base.SetEnabledStateOfControls();
|
---|
[6976] | 119 |
|
---|
[16122] | 120 | bool locked = Content == null || Locked || ReadOnly;
|
---|
| 121 | bool addLocked = locked
|
---|
| 122 | || !IsAdmin()
|
---|
| 123 | || (selectedResource is Slave && selectedResource.ParentResourceId != null)
|
---|
| 124 | || (selectedResource != null && selectedResource.Id == Guid.Empty);
|
---|
[6976] | 125 |
|
---|
[16122] | 126 | HashSet<Guid> descendantResources = null;
|
---|
| 127 |
|
---|
| 128 | bool deleteLocked = locked
|
---|
| 129 | || !IsAdmin()
|
---|
| 130 | || !Content.Any()
|
---|
| 131 | || selectedResource == null
|
---|
| 132 | || (selectedResource.Id != Guid.Empty && (!HiveAdminClient.Instance.ResourceDescendants.TryGetValue(selectedResource.Id, out descendantResources) || descendantResources.Any()));
|
---|
| 133 |
|
---|
| 134 | bool saveLocked = locked
|
---|
| 135 | || !IsAdmin()
|
---|
| 136 | || !Content.Any()
|
---|
| 137 | || selectedResource == null;
|
---|
| 138 |
|
---|
[16117] | 139 | btnAddGroup.Enabled = !addLocked;
|
---|
| 140 | btnRemoveGroup.Enabled = !deleteLocked;
|
---|
[16122] | 141 | btnSave.Enabled = !saveLocked;
|
---|
| 142 | viewHost.Locked = locked || !IsAdmin();
|
---|
| 143 | scheduleView.Locked = locked || !IsAdmin();
|
---|
[16117] | 144 | }
|
---|
| 145 | #endregion
|
---|
[6976] | 146 |
|
---|
[16117] | 147 | #region Event Handlers
|
---|
| 148 | private void Content_ItemsAdded(object sender, CollectionItemsChangedEventArgs<IndexedItem<Resource>> e) {
|
---|
| 149 | if (InvokeRequired) Invoke((Action<object, CollectionItemsChangedEventArgs<IndexedItem<Resource>>>)Content_ItemsAdded, sender, e);
|
---|
| 150 | else {
|
---|
| 151 | OnContentChanged();
|
---|
[6976] | 152 | }
|
---|
| 153 | }
|
---|
| 154 |
|
---|
[16117] | 155 | private void Content_ItemsRemoved(object sender, CollectionItemsChangedEventArgs<IndexedItem<Resource>> e) {
|
---|
| 156 | if (InvokeRequired) Invoke((Action<object, CollectionItemsChangedEventArgs<IndexedItem<Resource>>>)Content_ItemsRemoved, sender, e);
|
---|
| 157 | else {
|
---|
| 158 | OnContentChanged();
|
---|
| 159 | }
|
---|
| 160 | }
|
---|
| 161 |
|
---|
| 162 | private void SlaveViewContent_PropertyChanged(object sender, PropertyChangedEventArgs e) {
|
---|
| 163 | if (InvokeRequired) Invoke((Action<object, PropertyChangedEventArgs>)SlaveViewContent_PropertyChanged, sender, e);
|
---|
| 164 | else {
|
---|
| 165 | OnContentChanged();
|
---|
| 166 | if (e.PropertyName == "HbInterval") {
|
---|
| 167 | UpdateChildHbIntervall((Resource)viewHost.Content);
|
---|
[6976] | 168 | }
|
---|
[16117] | 169 | }
|
---|
| 170 | }
|
---|
[6976] | 171 |
|
---|
[16117] | 172 | private void HiveAdminClient_Instance_Refreshing(object sender, EventArgs e) {
|
---|
| 173 | if (InvokeRequired) Invoke((Action<object, EventArgs>)HiveAdminClient_Instance_Refreshing, sender, e);
|
---|
| 174 | else {
|
---|
| 175 | var mainForm = MainFormManager.GetMainForm<MainForm.WindowsForms.MainForm>();
|
---|
| 176 | mainForm.AddOperationProgressToView(this, "Refreshing ...");
|
---|
| 177 | SetEnabledStateOfControls();
|
---|
[6976] | 178 | }
|
---|
| 179 | }
|
---|
| 180 |
|
---|
[16117] | 181 | private void HiveAdminClient_Instance_Refreshed(object sender, EventArgs e) {
|
---|
| 182 | if (InvokeRequired) Invoke((Action<object, EventArgs>)HiveAdminClient_Instance_Refreshed, sender, e);
|
---|
| 183 | else {
|
---|
| 184 | var mainForm = MainFormManager.GetMainForm<MainForm.WindowsForms.MainForm>();
|
---|
| 185 | mainForm.RemoveOperationProgressFromView(this);
|
---|
| 186 | SetEnabledStateOfControls();
|
---|
[7250] | 187 | }
|
---|
[6976] | 188 | }
|
---|
| 189 |
|
---|
[16117] | 190 | private void AccessClient_Instance_Refreshing(object sender, EventArgs e) {
|
---|
| 191 | if (InvokeRequired) Invoke((Action<object, EventArgs>)AccessClient_Instance_Refreshing, sender, e);
|
---|
| 192 | else {
|
---|
| 193 | var mainForm = MainFormManager.GetMainForm<MainForm.WindowsForms.MainForm>();
|
---|
| 194 | mainForm.AddOperationProgressToView(this, "Refreshing ...");
|
---|
| 195 | SetEnabledStateOfControls();
|
---|
| 196 | }
|
---|
[8051] | 197 | }
|
---|
[6976] | 198 |
|
---|
[16117] | 199 | private void AccessClient_Instance_Refreshed(object sender, EventArgs e) {
|
---|
| 200 | if (InvokeRequired) Invoke((Action<object, EventArgs>)AccessClient_Instance_Refreshed, sender, e);
|
---|
| 201 | else {
|
---|
| 202 | var mainForm = MainFormManager.GetMainForm<MainForm.WindowsForms.MainForm>();
|
---|
| 203 | mainForm.RemoveOperationProgressFromView(this);
|
---|
| 204 | SetEnabledStateOfControls();
|
---|
| 205 | }
|
---|
| 206 | }
|
---|
[6976] | 207 |
|
---|
[16117] | 208 | private async void ResourcesView_Load(object sender, EventArgs e) {
|
---|
| 209 | await SecurityExceptionUtil.TryAsyncAndReportSecurityExceptions(
|
---|
| 210 | action: () => UpdateResources());
|
---|
| 211 | }
|
---|
[8051] | 212 |
|
---|
[16117] | 213 | private async void btnRefresh_Click(object sender, EventArgs e) {
|
---|
| 214 | lock (locker) {
|
---|
| 215 | if (!btnRefresh.Enabled) return;
|
---|
| 216 | btnRefresh.Enabled = false;
|
---|
| 217 | }
|
---|
[8051] | 218 |
|
---|
[16117] | 219 | await SecurityExceptionUtil.TryAsyncAndReportSecurityExceptions(
|
---|
| 220 | action: () => UpdateResources(),
|
---|
| 221 | finallyCallback: () => btnRefresh.Enabled = true);
|
---|
| 222 | }
|
---|
[8051] | 223 |
|
---|
[16117] | 224 | private void btnAddGroup_Click(object sender, EventArgs e) {
|
---|
[16122] | 225 | var parentResourceId = selectedResource is SlaveGroup ? selectedResource.Id : (Guid?)null;
|
---|
[16117] | 226 |
|
---|
| 227 | var group = new SlaveGroup {
|
---|
| 228 | Name = "New Group",
|
---|
| 229 | OwnerUserId = UserInformation.Instance.User.Id,
|
---|
| 230 | ParentResourceId = parentResourceId
|
---|
| 231 | };
|
---|
| 232 |
|
---|
| 233 | SelectedResource = group;
|
---|
| 234 | Content.Add(group);
|
---|
[6976] | 235 | }
|
---|
| 236 |
|
---|
[16117] | 237 | private async void btnRemoveGroup_Click(object sender, EventArgs e) {
|
---|
| 238 | if (selectedResource == null) return;
|
---|
| 239 |
|
---|
| 240 | lock (locker) {
|
---|
| 241 | if (!btnRemoveGroup.Enabled) return;
|
---|
| 242 | btnRemoveGroup.Enabled = false;
|
---|
[6976] | 243 | }
|
---|
[16117] | 244 |
|
---|
| 245 | var result = MessageBox.Show(
|
---|
| 246 | "Do you really want to delete " + selectedResource.Name + "?",
|
---|
| 247 | "HeuristicLab Hive Administrator",
|
---|
| 248 | MessageBoxButtons.YesNo,
|
---|
| 249 | MessageBoxIcon.Question);
|
---|
| 250 | if (result == DialogResult.Yes) {
|
---|
| 251 | await SecurityExceptionUtil.TryAsyncAndReportSecurityExceptions(
|
---|
| 252 | action: () => {
|
---|
| 253 | RemoveResource(selectedResource);
|
---|
| 254 | });
|
---|
[16122] | 255 |
|
---|
| 256 | OnContentChanged();
|
---|
| 257 | SetEnabledStateOfControls();
|
---|
[16117] | 258 | }
|
---|
[6976] | 259 | }
|
---|
| 260 |
|
---|
[16117] | 261 | private async void btnSave_Click(object sender, EventArgs e) {
|
---|
| 262 | lock (locker) {
|
---|
| 263 | if (!btnSave.Enabled) return;
|
---|
| 264 | btnSave.Enabled = false;
|
---|
[6976] | 265 | }
|
---|
[16117] | 266 |
|
---|
| 267 | await SecurityExceptionUtil.TryAsyncAndReportSecurityExceptions(
|
---|
| 268 | action: () => {
|
---|
| 269 | var resourcesToSave = Content.Where(x => x.Id == Guid.Empty || x.Modified);
|
---|
| 270 | foreach (var resource in resourcesToSave)
|
---|
| 271 | resource.Store();
|
---|
| 272 | UpdateResources();
|
---|
[16122] | 273 | });
|
---|
[16117] | 274 |
|
---|
| 275 | OnContentChanged();
|
---|
[16122] | 276 | SetEnabledStateOfControls();
|
---|
[6976] | 277 | }
|
---|
| 278 |
|
---|
[16117] | 279 | private void treeSlaveGroup_MouseDown(object sender, MouseEventArgs e) {
|
---|
| 280 | var node = treeView.GetNodeAt(e.Location);
|
---|
| 281 | if (node == null || node == ungroupedGroupNode) return;
|
---|
| 282 | var r = (Resource)node.Tag;
|
---|
| 283 | if (!HiveAdminClient.Instance.DisabledParentResources.Contains(r)) ChangeSelectedResourceNode(node);
|
---|
[6976] | 284 | }
|
---|
| 285 |
|
---|
[16117] | 286 | private void treeSlaveGroup_BeforeSelect(object sender, TreeViewCancelEventArgs e) {
|
---|
| 287 | e.Cancel = true;
|
---|
[6976] | 288 | }
|
---|
| 289 |
|
---|
[16117] | 290 | private void treeSlaveGroup_BeforeCheck(object sender, TreeViewCancelEventArgs e) {
|
---|
| 291 | if (e.Node == ungroupedGroupNode) {
|
---|
| 292 | e.Cancel = true;
|
---|
| 293 | } else {
|
---|
| 294 | var r = (Resource)e.Node.Tag;
|
---|
| 295 | if (HiveAdminClient.Instance.DisabledParentResources.Contains(r)) {
|
---|
| 296 | e.Cancel = true;
|
---|
| 297 | }
|
---|
| 298 | }
|
---|
| 299 | }
|
---|
| 300 |
|
---|
| 301 | private void treeSlaveGroup_DragDrop(object sender, DragEventArgs e) {
|
---|
| 302 | if (e.Effect == DragDropEffects.None) return;
|
---|
| 303 |
|
---|
| 304 | var targetNode = treeView.GetNodeAt(treeView.PointToClient(new Point(e.X, e.Y)));
|
---|
| 305 | var targetResource = (targetNode != null) ? (Resource)targetNode.Tag : null;
|
---|
| 306 | var resources = e.Data.GetData(HeuristicLab.Common.Constants.DragDropDataFormat) as IEnumerable<Resource>;
|
---|
| 307 |
|
---|
| 308 | foreach (var r in resources) {
|
---|
| 309 | r.ParentResourceId = targetResource != null ? targetResource.Id : (Guid?)null;
|
---|
| 310 | }
|
---|
| 311 |
|
---|
| 312 | // TODO
|
---|
| 313 | //HiveAdminClient.Instance.UpdateResourceGenealogy(Content);
|
---|
[6976] | 314 | OnContentChanged();
|
---|
| 315 | }
|
---|
| 316 |
|
---|
[16117] | 317 | private void treeSlaveGroup_ItemDrag(object sender, ItemDragEventArgs e) {
|
---|
| 318 | if (!IsAdmin()) return;
|
---|
[6976] | 319 |
|
---|
[16117] | 320 | var nodes = GetCheckedNodes(treeView.Nodes).ToList();
|
---|
| 321 | TreeNode sourceNode = (TreeNode)e.Item;
|
---|
| 322 | if (!sourceNode.Checked) nodes.Add(sourceNode);
|
---|
| 323 | nodes.Remove(ungroupedGroupNode);
|
---|
| 324 | ungroupedGroupNode.Checked = false;
|
---|
| 325 | var resources = nodes.Select(x => x.Tag).OfType<Resource>().ToList();
|
---|
| 326 |
|
---|
| 327 | if (resources.Count > 0) {
|
---|
| 328 | DataObject data = new DataObject();
|
---|
| 329 | data.SetData(HeuristicLab.Common.Constants.DragDropDataFormat, resources);
|
---|
| 330 | var action = DoDragDrop(data, DragDropEffects.Copy | DragDropEffects.Link | DragDropEffects.Move);
|
---|
| 331 | if (action.HasFlag(DragDropEffects.Move)) {
|
---|
| 332 | foreach (var node in nodes) node.Remove();
|
---|
| 333 | StyleTreeNode(ungroupedGroupNode, (Resource)ungroupedGroupNode.Tag, resources);
|
---|
[6976] | 334 | }
|
---|
| 335 | }
|
---|
| 336 | }
|
---|
| 337 |
|
---|
[16117] | 338 | private IEnumerable<TreeNode> GetCheckedNodes(TreeNodeCollection nodes) {
|
---|
| 339 | if (nodes != null) {
|
---|
| 340 | foreach (var node in nodes.OfType<TreeNode>()) {
|
---|
| 341 | if (node.Checked && node != ungroupedGroupNode) yield return node;
|
---|
| 342 | foreach (var child in GetCheckedNodes(node.Nodes))
|
---|
| 343 | yield return child;
|
---|
[6976] | 344 | }
|
---|
| 345 | }
|
---|
| 346 | }
|
---|
| 347 |
|
---|
[16117] | 348 | private void treeSlaveGroup_DragEnterOver(object sender, DragEventArgs e) {
|
---|
| 349 | e.Effect = DragDropEffects.Move;
|
---|
| 350 | var resources = e.Data.GetData(HeuristicLab.Common.Constants.DragDropDataFormat) as IEnumerable<Resource>;
|
---|
| 351 | var targetNode = treeView.GetNodeAt(treeView.PointToClient(new Point(e.X, e.Y)));
|
---|
| 352 | var targetResource = (targetNode != null ? targetNode.Tag : null) as Resource;
|
---|
[6976] | 353 |
|
---|
[16117] | 354 | if (!IsAdmin()
|
---|
| 355 | || resources == null
|
---|
| 356 | || !resources.Any()
|
---|
| 357 | || resources.Any(x => !HiveAdminClient.Instance.CheckParentChange(x, targetResource))
|
---|
| 358 | || (targetNode != null && (targetNode == ungroupedGroupNode || targetNode.Parent == ungroupedGroupNode))) {
|
---|
| 359 | e.Effect = DragDropEffects.None;
|
---|
| 360 | }
|
---|
| 361 | }
|
---|
[6976] | 362 |
|
---|
[16117] | 363 | private void TabSlaveGroup_TabIndexChanged(object sender, EventArgs e) {
|
---|
| 364 | throw new NotImplementedException();
|
---|
| 365 | }
|
---|
[6976] | 366 |
|
---|
[16117] | 367 | private async void TabSlaveGroup_Selected(object sender, System.Windows.Forms.TabControlEventArgs e) {
|
---|
| 368 | if (e.TabPage == tabSchedule) {
|
---|
| 369 | await SecurityExceptionUtil.TryAsyncAndReportSecurityExceptions(
|
---|
| 370 | action: () => UpdateSchedule(),
|
---|
| 371 | finallyCallback: () => scheduleView.Content = HiveAdminClient.Instance.Downtimes);
|
---|
| 372 | }
|
---|
| 373 | SetEnabledStateOfControls();
|
---|
| 374 | }
|
---|
| 375 | #endregion
|
---|
[6976] | 376 |
|
---|
[16117] | 377 | #region Helpers
|
---|
| 378 | private void BuildResourceTree(IEnumerable<Resource> resources) {
|
---|
| 379 | treeView.Nodes.Clear();
|
---|
| 380 | if (!resources.Any()) return;
|
---|
[8051] | 381 |
|
---|
[16117] | 382 | var disabledParentResources = HiveAdminClient.Instance.DisabledParentResources;
|
---|
| 383 | var mainResources = new HashSet<Resource>(resources.OfType<SlaveGroup>()
|
---|
| 384 | .Where(x => x.ParentResourceId == null));
|
---|
| 385 | //var parentedMainResources = new HashSet<Resource>(resources.OfType<SlaveGroup>()
|
---|
| 386 | // .Where(x => x.ParentResourceId.HasValue && !resources.Select(y => y.Id).Contains(x.ParentResourceId.Value)));
|
---|
| 387 | //mainResources.UnionWith(parentedMainResources);
|
---|
| 388 | var mainDisabledParentResources = new HashSet<Resource>(disabledParentResources.Where(x => x.ParentResourceId == null || x.ParentResourceId == Guid.Empty));
|
---|
| 389 | mainResources.UnionWith(mainDisabledParentResources);
|
---|
| 390 | var subResources = new HashSet<Resource>(resources.Union(disabledParentResources).Except(mainResources).OrderByDescending(x => x.Name));
|
---|
[8051] | 391 |
|
---|
[16117] | 392 | var stack = new Stack<Resource>(mainResources.OrderByDescending(x => x.Name));
|
---|
| 393 | if (selectedResource != null) SelectedResource = resources.Where(x => x.Id == selectedResource.Id).FirstOrDefault();
|
---|
| 394 | bool nodeSelected = false;
|
---|
[6976] | 395 |
|
---|
[16117] | 396 | TreeNode currentNode = null;
|
---|
| 397 | Resource currentResource = null;
|
---|
| 398 |
|
---|
| 399 | while (stack.Any()) {
|
---|
| 400 | var newResource = stack.Pop();
|
---|
| 401 | var newNode = new TreeNode(newResource.Name) { Tag = newResource };
|
---|
| 402 | StyleTreeNode(newNode, newResource, resources);
|
---|
| 403 |
|
---|
| 404 | if (selectedResource == null && !disabledParentResources.Contains(newResource)) {
|
---|
| 405 | SelectedResource = newResource;
|
---|
| 406 | }
|
---|
| 407 | if (!nodeSelected && selectedResource != null && newResource.Id == selectedResource.Id) {
|
---|
| 408 | newNode.BackColor = selectedBackColor;
|
---|
| 409 | newNode.ForeColor = selectedForeColor;
|
---|
| 410 | newNode.Text += SELECTED_TAG;
|
---|
| 411 | nodeSelected = true;
|
---|
| 412 | }
|
---|
| 413 |
|
---|
| 414 | if (disabledParentResources.Contains(newResource)) {
|
---|
| 415 | newNode.Checked = false;
|
---|
| 416 | newNode.ForeColor = grayTextColor;
|
---|
| 417 | }
|
---|
| 418 |
|
---|
| 419 | // search for parent node of newNode and save in currentNode
|
---|
| 420 | // necessary since newNodes (stack top items) might be siblings
|
---|
| 421 | // or grand..grandparents of previous node (currentNode)
|
---|
| 422 | while (currentNode != null && newResource.ParentResourceId != currentResource.Id) {
|
---|
| 423 | currentNode = currentNode.Parent;
|
---|
| 424 | currentResource = currentNode == null ? null : (Resource)currentNode.Tag;
|
---|
| 425 | }
|
---|
| 426 |
|
---|
| 427 | if (currentNode == null) {
|
---|
| 428 | treeView.Nodes.Add(newNode);
|
---|
| 429 | } else {
|
---|
| 430 | currentNode.Nodes.Add(newNode);
|
---|
| 431 | }
|
---|
| 432 |
|
---|
| 433 | if (newResource is SlaveGroup) {
|
---|
| 434 | var childResources = subResources.Where(x => x.ParentResourceId == newResource.Id);
|
---|
| 435 | if (childResources.Any()) {
|
---|
| 436 | foreach (var resource in childResources.OrderByDescending(x => x.Name)) {
|
---|
| 437 | subResources.Remove(resource);
|
---|
| 438 | stack.Push(resource);
|
---|
[6976] | 439 | }
|
---|
[16117] | 440 | currentNode = newNode;
|
---|
| 441 | currentResource = newResource;
|
---|
[6976] | 442 | }
|
---|
| 443 | }
|
---|
[16117] | 444 | newNode.SelectedImageIndex = newNode.ImageIndex;
|
---|
[6976] | 445 | }
|
---|
| 446 |
|
---|
[16117] | 447 | // collapse slave-only nodes
|
---|
| 448 | foreach (TreeNode n in treeView.Nodes) {
|
---|
| 449 | CollapseSlaveOnlyNodes(n);
|
---|
| 450 | }
|
---|
[6976] | 451 |
|
---|
[16117] | 452 | ungroupedGroupNode = new TreeNode(UNGROUPED_GROUP_NAME) {
|
---|
| 453 | ForeColor = SystemColors.GrayText,
|
---|
[16122] | 454 | ImageIndex = slaveGroupImageIndex,
|
---|
[16117] | 455 | Tag = new SlaveGroup() {
|
---|
| 456 | Name = UNGROUPED_GROUP_NAME,
|
---|
| 457 | Description = UNGROUPED_GROUP_DESCRIPTION
|
---|
[6976] | 458 | }
|
---|
[16117] | 459 | };
|
---|
| 460 |
|
---|
| 461 | foreach (var slave in subResources.OfType<Slave>().OrderBy(x => x.Name)) {
|
---|
| 462 | var slaveNode = new TreeNode(slave.Name) { Tag = slave };
|
---|
| 463 | StyleTreeNode(slaveNode, slave, resources);
|
---|
| 464 | ungroupedGroupNode.Nodes.Add(slaveNode);
|
---|
| 465 | if (selectedResource == null) {
|
---|
| 466 | SelectedResource = slave;
|
---|
| 467 | }
|
---|
| 468 |
|
---|
| 469 | if (slave.Id == selectedResource.Id && !nodeSelected) {
|
---|
| 470 | slaveNode.BackColor = selectedBackColor;
|
---|
| 471 | slaveNode.ForeColor = selectedForeColor;
|
---|
| 472 | slaveNode.Text += SELECTED_TAG;
|
---|
| 473 | nodeSelected = true;
|
---|
| 474 | }
|
---|
[6976] | 475 | }
|
---|
| 476 |
|
---|
[16117] | 477 | if (ungroupedGroupNode.Nodes.Count > 0) {
|
---|
| 478 | ungroupedGroupNode.Text += " [" + ungroupedGroupNode.Nodes.Count.ToString() + "]";
|
---|
| 479 | ungroupedGroupNode.Expand();
|
---|
[16122] | 480 | treeView.Nodes.Add(ungroupedGroupNode);
|
---|
[16117] | 481 | }
|
---|
[6976] | 482 | }
|
---|
| 483 |
|
---|
[16117] | 484 | private void CollapseSlaveOnlyNodes(TreeNode tn) {
|
---|
| 485 | Resource r = (Resource)tn.Tag;
|
---|
| 486 | var descendants = GetResourceDescendants();
|
---|
| 487 | if (descendants.ContainsKey(r.Id)) {
|
---|
| 488 | if (descendants[r.Id].OfType<SlaveGroup>().Any()) {
|
---|
| 489 | tn.Expand();
|
---|
| 490 | foreach (TreeNode n in tn.Nodes) CollapseSlaveOnlyNodes(n);
|
---|
| 491 | } else {
|
---|
| 492 | tn.Collapse();
|
---|
| 493 | }
|
---|
| 494 | }
|
---|
[6976] | 495 | }
|
---|
| 496 |
|
---|
[16117] | 497 | private void ExpandResourceNodesOfInterest(TreeNodeCollection nodes) {
|
---|
| 498 | foreach (TreeNode n in nodes) {
|
---|
| 499 | Resource r = (Resource)n.Tag;
|
---|
| 500 | if (n.Nodes.Count > 0) {
|
---|
| 501 | if (HiveAdminClient.Instance.GetAvailableResourceDescendants(r.Id).OfType<SlaveGroup>().Any()) {
|
---|
| 502 | n.Expand();
|
---|
| 503 | ExpandResourceNodesOfInterest(n.Nodes);
|
---|
| 504 | } else {
|
---|
| 505 | n.Collapse();
|
---|
| 506 | }
|
---|
| 507 | } else {
|
---|
| 508 | n.Collapse();
|
---|
| 509 | }
|
---|
| 510 | }
|
---|
[6976] | 511 | }
|
---|
| 512 |
|
---|
[16117] | 513 | private void UpdateChildHbIntervall(Resource resource) {
|
---|
| 514 | foreach (Resource r in Content.Where(x => x.ParentResourceId == resource.Id)) {
|
---|
| 515 | r.HbInterval = resource.HbInterval;
|
---|
| 516 | if (r is SlaveGroup) {
|
---|
| 517 | UpdateChildHbIntervall(r);
|
---|
[7256] | 518 | }
|
---|
[6976] | 519 | }
|
---|
| 520 | }
|
---|
| 521 |
|
---|
| 522 | private void UpdateResources() {
|
---|
| 523 | try {
|
---|
| 524 | HiveAdminClient.Instance.Refresh();
|
---|
| 525 | Content = HiveAdminClient.Instance.Resources;
|
---|
[16117] | 526 | } catch (AnonymousUserException) {
|
---|
| 527 | ShowHiveInformationDialog();
|
---|
[6976] | 528 | }
|
---|
[16117] | 529 | }
|
---|
| 530 |
|
---|
| 531 | private void RemoveResource(Resource resource) {
|
---|
| 532 | if (resource == null) return;
|
---|
| 533 |
|
---|
| 534 | try {
|
---|
| 535 | if (resource.Id != Guid.Empty) {
|
---|
| 536 | SelectedResource = HiveAdminClient.Instance.GetAvailableResourceAncestors(resource.Id).LastOrDefault();
|
---|
| 537 | HiveAdminClient.Delete(resource);
|
---|
| 538 | UpdateResources();
|
---|
| 539 | } else {
|
---|
[16122] | 540 | SelectedResource = null;
|
---|
| 541 | Content.Remove(resource);
|
---|
[16117] | 542 | }
|
---|
| 543 | } catch (AnonymousUserException) {
|
---|
| 544 | ShowHiveInformationDialog();
|
---|
[6976] | 545 | }
|
---|
[16117] | 546 | }
|
---|
| 547 |
|
---|
| 548 | private void UpdateSchedule() {
|
---|
| 549 | try {
|
---|
| 550 | HiveAdminClient.Instance.RefreshCalendar();
|
---|
| 551 | } catch (AnonymousUserException) {
|
---|
[7256] | 552 | ShowHiveInformationDialog();
|
---|
[7249] | 553 | }
|
---|
[6976] | 554 | }
|
---|
| 555 |
|
---|
[16117] | 556 | private bool IsAdmin() {
|
---|
| 557 | return HiveRoles.CheckAdminUserPermissions();
|
---|
| 558 | }
|
---|
| 559 |
|
---|
| 560 | private void StyleTreeNode(TreeNode n, Resource r, IEnumerable<Resource> resources) {
|
---|
| 561 | n.Text = r.Name;
|
---|
| 562 | n.BackColor = Color.Transparent;
|
---|
| 563 | n.ForeColor = Color.Black;
|
---|
| 564 |
|
---|
| 565 | if (HiveAdminClient.Instance.DisabledParentResources.Select(x => x.Id).Contains(r.Id)) {
|
---|
| 566 | n.ForeColor = grayTextColor;
|
---|
| 567 | } else if (r.Id == Guid.Empty && n != ungroupedGroupNode /*!r.Name.StartsWith(UNGROUPED_GROUP_NAME)*/) {
|
---|
| 568 | // not stored (i.e. new)
|
---|
| 569 | n.Text += NOT_STORED_TAG;
|
---|
| 570 | } else if (r.Modified && n != ungroupedGroupNode /*!r.Name.StartsWith(UNGROUPED_GROUP_NAME)*/) {
|
---|
| 571 | // changed
|
---|
| 572 | n.Text += CHANGES_NOT_STORED_TAG;
|
---|
| 573 | }
|
---|
| 574 |
|
---|
| 575 | // slave count
|
---|
| 576 | int childSlavesCount = 0;
|
---|
| 577 | if (r.Id != Guid.Empty && r is SlaveGroup) {
|
---|
| 578 | var descendants = GetResourceDescendants();
|
---|
| 579 | if (descendants.ContainsKey(r.Id)) {
|
---|
| 580 | childSlavesCount = resources
|
---|
| 581 | .OfType<Slave>()
|
---|
| 582 | .Where(x => descendants[r.Id].Select(y => y.Id)
|
---|
| 583 | .Contains(x.Id))
|
---|
| 584 | .Count();
|
---|
| 585 | }
|
---|
| 586 | } else if (n == ungroupedGroupNode /*|| r.Name.StartsWith(UNGROUPED_GROUP_NAME)*/) {
|
---|
| 587 | childSlavesCount = resources
|
---|
| 588 | .OfType<Slave>()
|
---|
| 589 | .Where(x => x.ParentResourceId == null
|
---|
| 590 | || (x.ParentResourceId.HasValue && x.ParentResourceId.Value == Guid.Empty))
|
---|
| 591 | .Count();
|
---|
| 592 | }
|
---|
| 593 | if (childSlavesCount > 0)
|
---|
| 594 | n.Text += " [" + childSlavesCount.ToString() + "]";
|
---|
| 595 |
|
---|
| 596 | // slave image index, state, utilization
|
---|
| 597 | if (r is Slave) {
|
---|
| 598 | n.ImageIndex = slaveImageIndex;
|
---|
| 599 | var s = r as Slave;
|
---|
| 600 | if (s.SlaveState == SlaveState.Calculating) {
|
---|
| 601 | n.ForeColor = calculatingColor;
|
---|
| 602 | n.Text += " [" + s.CpuUtilization.ToString("N2") + "%]";
|
---|
| 603 | } else if (s.SlaveState == SlaveState.Offline) {
|
---|
| 604 | n.ForeColor = offlineColor;
|
---|
| 605 | if (s.LastHeartbeat.HasValue)
|
---|
| 606 | n.Text += " [" + (s.LastHeartbeat != null ? s.LastHeartbeat.Value.ToString("g") : null) + "]";
|
---|
| 607 | }
|
---|
[7256] | 608 | } else {
|
---|
[16117] | 609 | n.ImageIndex = slaveGroupImageIndex;
|
---|
[7256] | 610 | }
|
---|
[16117] | 611 |
|
---|
| 612 | // ungrouped
|
---|
| 613 | if (n == ungroupedGroupNode /*r.Name.StartsWith(UNGROUPED_GROUP_NAME)*/) {
|
---|
| 614 | n.ForeColor = SystemColors.GrayText;
|
---|
| 615 | }
|
---|
[7256] | 616 | }
|
---|
| 617 |
|
---|
[16117] | 618 | private void ResetTreeNodes(TreeNodeCollection nodes, IEnumerable<Resource> resources) {
|
---|
| 619 | foreach (TreeNode n in nodes) {
|
---|
| 620 | StyleTreeNode(n, (Resource)n.Tag, resources);
|
---|
| 621 | if (n.Nodes.Count > 0) {
|
---|
| 622 | ResetTreeNodes(n.Nodes, resources);
|
---|
[7256] | 623 | }
|
---|
| 624 | }
|
---|
| 625 | }
|
---|
| 626 |
|
---|
[16117] | 627 | private async void ChangeSelectedResource(Resource resource) {
|
---|
| 628 | selectedResource = resource;
|
---|
| 629 | viewHost.Content = selectedResource;
|
---|
[6976] | 630 |
|
---|
[16117] | 631 | HiveAdminClient.Instance.DowntimeForResourceId = selectedResource != null ? selectedResource.Id : Guid.Empty;
|
---|
| 632 | if (tabSlaveGroup.SelectedTab == tabSchedule) {
|
---|
| 633 | await SecurityExceptionUtil.TryAsyncAndReportSecurityExceptions(
|
---|
| 634 | action: () => UpdateSchedule(),
|
---|
| 635 | finallyCallback: () => scheduleView.Content = HiveAdminClient.Instance.Downtimes);
|
---|
| 636 | }
|
---|
[6976] | 637 |
|
---|
[16117] | 638 | SetEnabledStateOfControls();
|
---|
[6976] | 639 | }
|
---|
| 640 |
|
---|
[16117] | 641 | private void ChangeSelectedResourceNode(TreeNode resourceNode) {
|
---|
| 642 | if (resourceNode == null) return;
|
---|
| 643 | SelectedResource = (Resource)resourceNode.Tag;
|
---|
| 644 | ResetTreeNodes(treeView.Nodes, Content);
|
---|
| 645 | resourceNode.BackColor = selectedBackColor;
|
---|
| 646 | resourceNode.ForeColor = selectedForeColor;
|
---|
| 647 | resourceNode.Text += SELECTED_TAG;
|
---|
[8051] | 648 | }
|
---|
| 649 |
|
---|
[16117] | 650 | private void ShowHiveInformationDialog() {
|
---|
| 651 | if (InvokeRequired) Invoke((Action)ShowHiveInformationDialog);
|
---|
| 652 | else {
|
---|
| 653 | using (HiveInformationDialog dialog = new HiveInformationDialog()) {
|
---|
| 654 | dialog.ShowDialog(this);
|
---|
| 655 | }
|
---|
| 656 | }
|
---|
[8051] | 657 | }
|
---|
| 658 |
|
---|
[16117] | 659 | private void ResetView() {
|
---|
| 660 | if (InvokeRequired) Invoke((Action)ResetView);
|
---|
| 661 | else {
|
---|
| 662 | treeView.Nodes.Clear();
|
---|
[8051] | 663 |
|
---|
[16117] | 664 | if (viewHost.Content != null && viewHost.Content is SlaveGroup) {
|
---|
| 665 | ((SlaveGroup)viewHost.Content).PropertyChanged -= SlaveViewContent_PropertyChanged;
|
---|
| 666 | }
|
---|
[6976] | 667 |
|
---|
[16117] | 668 | viewHost.Content = null;
|
---|
| 669 | if (scheduleView.Content != null) {
|
---|
| 670 | scheduleView.Content.Clear();
|
---|
| 671 | }
|
---|
| 672 |
|
---|
| 673 | HiveAdminClient.Instance.ResetDowntime();
|
---|
[6976] | 674 | }
|
---|
| 675 | }
|
---|
| 676 |
|
---|
[6994] | 677 |
|
---|
[16117] | 678 | private Dictionary<Guid, HashSet<Resource>> GetResourceDescendants() {
|
---|
| 679 | var resourceDescendants = new Dictionary<Guid, HashSet<Resource>>();
|
---|
| 680 | var resources = Content.Union(HiveAdminClient.Instance.DisabledParentResources).ToList();
|
---|
| 681 |
|
---|
| 682 | foreach (var r in resources) resourceDescendants.Add(r.Id, new HashSet<Resource>());
|
---|
| 683 | foreach (var r in resources) {
|
---|
| 684 | var parentResourceId = r.ParentResourceId;
|
---|
| 685 | while (parentResourceId != null) {
|
---|
| 686 | var parent = resources.SingleOrDefault(x => x.Id == parentResourceId);
|
---|
| 687 | if (parent != null) {
|
---|
| 688 | resourceDescendants[parent.Id].Add(r);
|
---|
| 689 | parentResourceId = parent.ParentResourceId;
|
---|
| 690 | } else {
|
---|
| 691 | parentResourceId = null;
|
---|
| 692 | }
|
---|
| 693 | }
|
---|
| 694 | }
|
---|
| 695 | return resourceDescendants;
|
---|
[6994] | 696 | }
|
---|
[8051] | 697 |
|
---|
[16117] | 698 | #endregion
|
---|
[6976] | 699 | }
|
---|
| 700 | }
|
---|