Browse Source

Initial commit with little error handling

Just a proof of concept at this moment. It works but may be buggy as
hell.
Bryan Allred 10 years ago
commit
4f8537caa9
12 changed files with 1293 additions and 0 deletions
  1. 22 0
      .gitattributes
  2. 215 0
      .gitignore
  3. 6 0
      App.config
  4. 556 0
      Dashboard.cs
  5. 18 0
      Penalties.cs
  6. 218 0
      Program.cs
  7. 36 0
      Properties/AssemblyInfo.cs
  8. 30 0
      README
  9. 53 0
      Rewards.cs
  10. 42 0
      Sound.cs
  11. 34 0
      TfsCredentialsProvider.cs
  12. 63 0
      pulse.csproj

+ 22 - 0
.gitattributes

@ -0,0 +1,22 @@
1
# Auto detect text files and perform LF normalization
2
* text=auto
3
4
# Custom for Visual Studio
5
*.cs     diff=csharp
6
*.sln    merge=union
7
*.csproj merge=union
8
*.vbproj merge=union
9
*.fsproj merge=union
10
*.dbproj merge=union
11
12
# Standard to msysgit
13
*.doc	 diff=astextplain
14
*.DOC	 diff=astextplain
15
*.docx diff=astextplain
16
*.DOCX diff=astextplain
17
*.dot  diff=astextplain
18
*.DOT  diff=astextplain
19
*.pdf  diff=astextplain
20
*.PDF	 diff=astextplain
21
*.rtf	 diff=astextplain
22
*.RTF	 diff=astextplain

+ 215 - 0
.gitignore

@ -0,0 +1,215 @@
1
#################
2
## Eclipse
3
#################
4
5
*.pydevproject
6
.project
7
.metadata
8
bin/
9
tmp/
10
*.tmp
11
*.bak
12
*.swp
13
*~.nib
14
local.properties
15
.classpath
16
.settings/
17
.loadpath
18
19
# External tool builders
20
.externalToolBuilders/
21
22
# Locally stored "Eclipse launch configurations"
23
*.launch
24
25
# CDT-specific
26
.cproject
27
28
# PDT-specific
29
.buildpath
30
31
32
#################
33
## Visual Studio
34
#################
35
36
## Ignore Visual Studio temporary files, build results, and
37
## files generated by popular Visual Studio add-ons.
38
39
# User-specific files
40
*.suo
41
*.user
42
*.sln.docstates
43
44
# Build results
45
46
[Dd]ebug/
47
[Rr]elease/
48
x64/
49
build/
50
[Bb]in/
51
[Oo]bj/
52
53
# MSTest test Results
54
[Tt]est[Rr]esult*/
55
[Bb]uild[Ll]og.*
56
57
*_i.c
58
*_p.c
59
*.ilk
60
*.meta
61
*.obj
62
*.pch
63
*.pdb
64
*.pgc
65
*.pgd
66
*.rsp
67
*.sbr
68
*.tlb
69
*.tli
70
*.tlh
71
*.tmp
72
*.tmp_proj
73
*.log
74
*.vspscc
75
*.vssscc
76
.builds
77
*.pidb
78
*.log
79
*.scc
80
81
# Visual C++ cache files
82
ipch/
83
*.aps
84
*.ncb
85
*.opensdf
86
*.sdf
87
*.cachefile
88
89
# Visual Studio profiler
90
*.psess
91
*.vsp
92
*.vspx
93
94
# Guidance Automation Toolkit
95
*.gpState
96
97
# ReSharper is a .NET coding add-in
98
_ReSharper*/
99
*.[Rr]e[Ss]harper
100
101
# TeamCity is a build add-in
102
_TeamCity*
103
104
# DotCover is a Code Coverage Tool
105
*.dotCover
106
107
# NCrunch
108
*.ncrunch*
109
.*crunch*.local.xml
110
111
# Installshield output folder
112
[Ee]xpress/
113
114
# DocProject is a documentation generator add-in
115
DocProject/buildhelp/
116
DocProject/Help/*.HxT
117
DocProject/Help/*.HxC
118
DocProject/Help/*.hhc
119
DocProject/Help/*.hhk
120
DocProject/Help/*.hhp
121
DocProject/Help/Html2
122
DocProject/Help/html
123
124
# Click-Once directory
125
publish/
126
127
# Publish Web Output
128
*.Publish.xml
129
*.pubxml
130
131
# NuGet Packages Directory
132
## TODO: If you have NuGet Package Restore enabled, uncomment the next line
133
#packages/
134
135
# Windows Azure Build Output
136
csx
137
*.build.csdef
138
139
# Windows Store app package directory
140
AppPackages/
141
142
# Others
143
sql/
144
*.Cache
145
ClientBin/
146
[Ss]tyle[Cc]op.*
147
~$*
148
*~
149
*.dbmdl
150
*.[Pp]ublish.xml
151
*.pfx
152
*.publishsettings
153
154
# RIA/Silverlight projects
155
Generated_Code/
156
157
# Backup & report files from converting an old project file to a newer
158
# Visual Studio version. Backup files are not needed, because we have git ;-)
159
_UpgradeReport_Files/
160
Backup*/
161
UpgradeLog*.XML
162
UpgradeLog*.htm
163
164
# SQL Server files
165
App_Data/*.mdf
166
App_Data/*.ldf
167
168
#############
169
## Windows detritus
170
#############
171
172
# Windows image file caches
173
Thumbs.db
174
ehthumbs.db
175
176
# Folder config file
177
Desktop.ini
178
179
# Recycle Bin used on file shares
180
$RECYCLE.BIN/
181
182
# Mac crap
183
.DS_Store
184
185
186
#############
187
## Python
188
#############
189
190
*.py[co]
191
192
# Packages
193
*.egg
194
*.egg-info
195
dist/
196
build/
197
eggs/
198
parts/
199
var/
200
sdist/
201
develop-eggs/
202
.installed.cfg
203
204
# Installer logs
205
pip-log.txt
206
207
# Unit test / coverage reports
208
.coverage
209
.tox
210
211
#Translations
212
*.mo
213
214
#Mr Developer
215
.mr.developer.cfg

+ 6 - 0
App.config

@ -0,0 +1,6 @@
1
<?xml version="1.0" encoding="utf-8" ?>
2
<configuration>
3
    <startup> 
4
        <supportedRuntime version="v4.0" sku=".NETFramework,Version=v4.5" />
5
    </startup>
6
</configuration>

+ 556 - 0
Dashboard.cs

@ -0,0 +1,556 @@
1
using System;
2
using System.Collections.Generic;
3
using System.Diagnostics;
4
using System.IO;
5
using System.Linq;
6
using Microsoft.TeamFoundation.VersionControl.Client;
7
using Microsoft.TeamFoundation.WorkItemTracking.Client;
8
9
namespace pulse
10
{
11
	/// <summary>
12
	/// Dashboard.
13
	/// </summary>
14
	public class Dashboard
15
	{
16
		/// <summary>
17
		/// Gets or sets the server.
18
		/// </summary>
19
		/// <value>
20
		/// The server.
21
		/// </value>
22
		public Uri Server { get; set; }
23
24
		/// <summary>
25
		/// Gets or sets the selected item.
26
		/// </summary>
27
		/// <value>
28
		/// The selected item.
29
		/// </value>
30
		public int SelectedItem { get; set; }
31
32
		/// <summary>
33
		/// Gets or sets the date refresh.
34
		/// </summary>
35
		/// <value>
36
		/// The date refresh.
37
		/// </value>
38
		public DateTime? DateRefresh { get; set; }
39
40
		/// <summary>
41
		/// Gets or sets the scores.
42
		/// </summary>
43
		/// <value>
44
		/// The scores.
45
		/// </value>
46
		private IDictionary<string, double> Scores { get; set; }
47
48
		/// <summary>
49
		/// Gets or sets the highlighted scores.
50
		/// </summary>
51
		/// <value>
52
		/// The highlighted scores.
53
		/// </value>
54
		private IList<string> HighlightedScores { get; set; }
55
56
		/// <summary>
57
		/// Gets the cache file.
58
		/// </summary>
59
		/// <value>
60
		/// The cache file.
61
		/// </value>
62
		private string CacheFile
63
		{
64
			get
65
			{
66
				if (this.Server == null)
67
				{
68
					return "$cache";
69
				}
70
71
				return string.Format(
72
					"{0}_{1}_{2}.cache",
73
					this.Server.Scheme,
74
					this.Server.Host,
75
					this.Server.Port);
76
			}
77
		}
78
79
		/// <summary>
80
		/// Gets or sets the commits.
81
		/// </summary>
82
		/// <value>
83
		/// The commits.
84
		/// </value>
85
		private int Commits { get; set; }
86
87
		/// <summary>
88
		/// Gets or sets the projects.
89
		/// </summary>
90
		/// <value>
91
		/// The projects.
92
		/// </value>
93
		private int Projects { get; set; }
94
95
		/// <summary>
96
		/// Gets or sets the work items.
97
		/// </summary>
98
		/// <value>
99
		/// The work items.
100
		/// </value>
101
		private int WorkItems { get; set; }
102
103
		/// <summary>
104
		/// Gets the height.
105
		/// </summary>
106
		/// <value>
107
		/// The height.
108
		/// </value>
109
		private int Height
110
		{
111
			get
112
			{
113
				return Console.WindowHeight - 2;
114
			}
115
		}
116
117
		/// <summary>
118
		/// Gets the width.
119
		/// </summary>
120
		/// <value>
121
		/// The width.
122
		/// </value>
123
		private int Width
124
		{
125
			get
126
			{
127
				return Console.WindowWidth;
128
			}
129
		}
130
131
		/// <summary>
132
		/// Initializes a new instance of the <see cref="Dashboard" /> class.
133
		/// </summary>
134
		/// <param name="server">The server.</param>
135
		public Dashboard(Uri server)
136
		{
137
			this.Commits = 0;
138
			this.Projects = 0;
139
			this.WorkItems = 0;
140
			this.DateRefresh = null;
141
			this.Server = server;
142
			this.SelectedItem = 1;
143
			this.Scores = new Dictionary<string, double>();
144
			this.HighlightedScores = new List<string>();
145
146
			var cacheFile = new FileInfo(this.CacheFile);
147
148
			if (cacheFile.Exists)
149
			{
150
				using (var reader = new StreamReader(cacheFile.Open(FileMode.Open, FileAccess.Read, FileShare.Read)))
151
				{
152
					while (!reader.EndOfStream)
153
					{
154
						var line = reader.ReadLine();
155
						var fields = line.Split(new char[] { ';' });
156
157
						switch (fields[0].ToLower())
158
						{
159
							case "refreshed":
160
								if (fields.Count() > 1 && !string.IsNullOrWhiteSpace(fields[1]))
161
								{
162
									this.DateRefresh = Convert.ToDateTime(fields[1]);
163
								}
164
								break;
165
166
							case "projects":
167
								this.Projects = Convert.ToInt32(fields[1]);
168
								break;
169
170
							case "work":
171
								this.WorkItems = Convert.ToInt32(fields[1]);
172
								break;
173
174
							case "commits":
175
								this.Commits = Convert.ToInt32(fields[1]);
176
								break;
177
178
							case "score":
179
								var points = Convert.ToDouble(fields[2]);
180
181
								if (!this.Scores.ContainsKey(fields[1]))
182
								{
183
									this.Scores.Add(fields[1], points);
184
								}
185
								else
186
								{
187
									this.Scores[fields[1]] = points;
188
								}
189
190
								break;
191
						}
192
					}
193
				}
194
			}
195
		}
196
197
		/// <summary>
198
		/// Adds the commits.
199
		/// </summary>
200
		/// <param name="commits">The commits.</param>
201
		public void AddCommits(IEnumerable<Changeset> commits)
202
		{
203
			foreach (var commit in commits)
204
			{
205
				// Add to the overall tally.
206
				this.Commits++;
207
208
				var personOfInterest = commit.CommitterDisplayName;
209
				this.AddHighlightedScore(personOfInterest);
210
211
				if (!this.Scores.Any(x => x.Key.Equals(personOfInterest, StringComparison.InvariantCultureIgnoreCase)))
212
				{
213
					this.Scores.Add(personOfInterest, 0);
214
				}
215
216
				// Get a point for participation.
217
				this.Scores[personOfInterest] += Rewards.Participation;
218
219
				// Handle points for commits with comments.
220
				if (string.IsNullOrWhiteSpace(commit.Comment))
221
				{
222
					this.Scores[personOfInterest] += Penalties.LackingVerbosity;
223
				}
224
				else
225
				{
226
					this.Scores[personOfInterest] += Rewards.Verbosity;
227
				}
228
229
				// Handle points with associating commits with actual work.
230
				if (commit.AssociatedWorkItems.Count() == 0)
231
				{
232
					this.Scores[personOfInterest] += Penalties.LackOfPurpose;
233
				}
234
				else
235
				{
236
					this.Scores[personOfInterest] += Rewards.Collaboration;
237
				}
238
			}
239
		}
240
241
		/// <summary>
242
		/// Adds the projects.
243
		/// </summary>
244
		/// <param name="projects">The projects.</param>
245
		public void AddProjects(IEnumerable<object> projects)
246
		{
247
			this.Projects = projects.Count();
248
		}
249
250
		/// <summary>
251
		/// Adds the work items.
252
		/// </summary>
253
		/// <param name="workItems">The work items.</param>
254
		public void AddWorkItems(IEnumerable<WorkItem> workItems)
255
		{
256
			foreach (var item in workItems)
257
			{
258
				// Add to the overall tally.
259
				this.WorkItems++;
260
261
				if (!string.IsNullOrWhiteSpace(item.CreatedBy))
262
				{
263
					if (!this.Scores.Any(x => x.Key.Equals(item.CreatedBy, StringComparison.InvariantCultureIgnoreCase)))
264
					{
265
						this.Scores.Add(item.CreatedBy, 0);
266
					}
267
				}
268
269
				if (item.CreatedBy != item.ChangedBy)
270
				{
271
					if (!string.IsNullOrWhiteSpace(item.ChangedBy))
272
					{
273
						if (!this.Scores.Any(x => x.Key.Equals(item.ChangedBy, StringComparison.InvariantCultureIgnoreCase)))
274
						{
275
							this.Scores.Add(item.ChangedBy, 0);
276
						}
277
278
						// Add points for collaboration.
279
						this.Scores[item.ChangedBy] += Rewards.Collaboration;
280
					}
281
				}
282
				else
283
				{
284
					if (!string.IsNullOrWhiteSpace(item.CreatedBy))
285
					{
286
						// They must have updated something right?
287
						this.Scores[item.CreatedBy] += Rewards.Participation;
288
					}
289
				}
290
291
				var personOfInterest = item.CreatedDate < item.ChangedDate
292
					? item.ChangedBy
293
					: item.CreatedBy;
294
				this.AddHighlightedScore(personOfInterest);
295
296
				if (!string.IsNullOrWhiteSpace(personOfInterest))
297
				{
298
					switch (item.Type.Name.ToLower())
299
					{
300
						case "product backlog item":
301
							this.Scores[personOfInterest] += Rewards.Participation;
302
							break;
303
304
						case "user story":
305
							this.Scores[personOfInterest] += Rewards.Participation;
306
							break;
307
308
						case "feature":
309
							this.Scores[personOfInterest] += Rewards.Participation;
310
							break;
311
312
						case "task":
313
							this.Scores[personOfInterest] += Rewards.CreatedWork;
314
							break;
315
316
						case "impediment":
317
						case "issue":
318
						case "bug":
319
							this.Scores[personOfInterest] += Rewards.BugReport;
320
							break;
321
322
						case "test case":
323
						case "shared steps":
324
							this.Scores[personOfInterest] += Rewards.CreatedWork;
325
							break;
326
327
						default:
328
							Debug.WriteLine("Did not handle work item type [" + item.Type.Name + "]");
329
							break;
330
					}
331
332
					switch (item.State.ToLower())
333
					{
334
						case "new":
335
						case "design":
336
						case "to do":
337
						case "open":
338
						case "approved":
339
						case "active":
340
						case "in progress":
341
							this.Scores[personOfInterest] += Rewards.TakingOwnership;
342
							break;
343
344
						case "done":
345
						case "committed":
346
						case "resolved":
347
						case "closed":
348
							this.Scores[personOfInterest] += Rewards.FinishWork;
349
							break;
350
351
						case "removed":
352
							this.Scores[personOfInterest] += Rewards.HouseCleaning;
353
							break;
354
355
						default:
356
							Debug.WriteLine("Did not handle work item state [" + item.State + "]");
357
							break;
358
					}
359
				}
360
			}
361
		}
362
363
		/// <summary>
364
		/// Caches this instance.
365
		/// </summary>
366
		public void Cache()
367
		{
368
			var cacheFile = new FileInfo(this.CacheFile);
369
370
			using (var writer = new StreamWriter(cacheFile.Open(FileMode.OpenOrCreate, FileAccess.Write, FileShare.None)))
371
			{
372
				writer.WriteLine(string.Format("{0};{1}", "refreshed", this.DateRefresh.HasValue ? this.DateRefresh.Value.ToString() : string.Empty));
373
				writer.WriteLine(string.Format("{0};{1}", "projects", this.Projects));
374
				writer.WriteLine(string.Format("{0};{1}", "work", this.WorkItems));
375
				writer.WriteLine(string.Format("{0};{1}", "commits", this.Commits));
376
377
				foreach (var score in this.Scores)
378
				{
379
					writer.WriteLine(string.Format("{0};{1};{2}", "score", score.Key, score.Value));
380
				}
381
			}
382
		}
383
384
		/// <summary>
385
		/// Clears the highlights.
386
		/// </summary>
387
		public void ClearHighlights()
388
		{
389
			this.HighlightedScores.Clear();
390
		}
391
392
		/// <summary>
393
		/// Updates the specified refreshing.
394
		/// </summary>
395
		public void Update()
396
		{
397
			// Header
398
			var buffer = new List<string>();
399
			buffer.AddRange(this.Statistics());
400
401
			// Divider
402
			buffer.Add(Padding(this.Width, "-"));
403
404
			// Leaderboard
405
			var leaderBoard = this.Leadboard();
406
			var pageSize = this.Height - buffer.Count();
407
408
			// Reset the selected item if it exceeds the leader board limitations.
409
			if (this.SelectedItem > leaderBoard.Count())
410
			{
411
				this.SelectedItem = leaderBoard.Count();
412
			}
413
414
			// Figure out what to put in the page.
415
			if (this.SelectedItem > pageSize)
416
			{
417
				buffer.AddRange(leaderBoard.Skip(this.SelectedItem % pageSize).Take(pageSize));
418
			}
419
			else
420
			{
421
				buffer.AddRange(leaderBoard.Take(pageSize));
422
			}
423
424
			// Filler
425
			var heightRemaining = this.Height - buffer.Count();
426
			for (var i = 0; i < heightRemaining; i++)
427
			{
428
				buffer.Add(string.Empty);
429
			}
430
431
			Console.CursorVisible = false;
432
			Console.SetCursorPosition(0, 0);
433
			foreach (var line in buffer)
434
			{
435
				foreach (var highlight in this.HighlightedScores)
436
				{
437
					if (line.Contains(highlight))
438
					{
439
						Console.ForegroundColor = ConsoleColor.Green;
440
						break;
441
					}
442
				}
443
444
				if (line.StartsWith(string.Format(" {0,3:###}.", this.SelectedItem)))
445
				{
446
					Console.BackgroundColor = ConsoleColor.White;
447
					if (Console.ForegroundColor != ConsoleColor.Green)
448
					{
449
						Console.ForegroundColor = ConsoleColor.Black;
450
					}
451
				}
452
453
				if (line.Length >= this.Width)
454
				{
455
					Console.Write(line);
456
				}
457
				else
458
				{
459
					Console.WriteLine(line);
460
				}
461
462
				Console.ResetColor();
463
			}
464
465
			// Notify with a sound
466
			if (this.HighlightedScores.Count() > 0)
467
			{
468
				Sound.Queue.Enqueue(Sound.Notify);
469
			}
470
		}
471
472
		/// <summary>
473
		/// Adds the highlighted score.
474
		/// </summary>
475
		/// <param name="person">The person.</param>
476
		private void AddHighlightedScore(string person)
477
		{
478
			if (this.Scores.ContainsKey(person) && !this.HighlightedScores.Contains(person))
479
			{
480
				this.HighlightedScores.Add(person);
481
			}
482
		}
483
484
		/// <summary>
485
		/// Statisticses this instance.
486
		/// </summary>
487
		/// <returns></returns>
488
		private IEnumerable<string> Statistics()
489
		{
490
			var lines = new List<string>();
491
492
			var title = "pulse";
493
			var titlePadding = Padding((this.Width - title.Length) / 2);
494
			lines.Add(string.Format("{0}{1}{0}", titlePadding, title));
495
496
			var stats = string.Format(
497
				"p: {0} | w: {1} | c: {2} | r: {3:MM/dd/yyyy HH:mm}",
498
				this.Projects,
499
				this.WorkItems,
500
				this.Commits,
501
				this.DateRefresh.HasValue ? this.DateRefresh.Value : DateTime.Now);
502
			var statsPadding = Padding((this.Width - stats.Length) / 2);
503
			lines.Add(string.Format("{0}{1}{0}", statsPadding, stats));
504
505
			return lines;
506
		}
507
508
		/// <summary>
509
		/// Leadboards this instance.
510
		/// </summary>
511
		/// <returns></returns>
512
		private IEnumerable<string> Leadboard()
513
		{
514
			if (this.Scores == null || this.Scores.Count() == 0)
515
			{
516
				return new List<string>();
517
			}
518
519
			var rank = 0;
520
			return this.Scores
521
				.OrderByDescending(x => x.Value)
522
				.Select(x =>
523
				{
524
					var player = x.Key;
525
526
					var points = (int)x.Value;
527
					if (points < 0)
528
					{
529
						points = 0;
530
					}
531
532
					var basic = string.Format(" {0,3:###}. {1}{{0}}{2}", ++rank, player, points);
533
					return basic.Replace("{0}", Padding(this.Width - basic.Length + 2, " "));
534
				})
535
				.ToList();
536
		}
537
538
		/// <summary>
539
		/// Paddings the specified length.
540
		/// </summary>
541
		/// <param name="length">The length.</param>
542
		/// <param name="text">The text.</param>
543
		/// <returns></returns>
544
		private static string Padding(int length, string text = " ")
545
		{
546
			var padding = string.Empty;
547
548
			for (var i = 0; i < length; i++)
549
			{
550
				padding += text;
551
			}
552
553
			return padding;
554
		}
555
	}
556
}

+ 18 - 0
Penalties.cs

@ -0,0 +1,18 @@
1
namespace pulse
2
{
3
	/// <summary>
4
	/// Penalty point values.
5
	/// </summary>
6
	public static class Penalties
7
	{
8
		/// <summary>
9
		/// The lacking verbosity.
10
		/// </summary>
11
		public const double LackingVerbosity = -0.2;
12
13
		/// <summary>
14
		/// The lack of purpose.
15
		/// </summary>
16
		public const double LackOfPurpose = -0.2;
17
	}
18
}

+ 218 - 0
Program.cs

@ -0,0 +1,218 @@
1
using System;
2
using System.ComponentModel;
3
using System.Diagnostics;
4
using System.Linq;
5
using System.Threading;
6
using Microsoft.TeamFoundation.Client;
7
using Microsoft.TeamFoundation.Framework.Common;
8
using Microsoft.TeamFoundation.VersionControl.Client;
9
using Microsoft.TeamFoundation.WorkItemTracking.Client;
10
11
namespace pulse
12
{
13
	/// <summary>
14
	/// The program.
15
	/// </summary>
16
	public class Program
17
	{
18
		/// <summary>
19
		/// Mains the specified arguments.
20
		/// </summary>
21
		/// <param name="args">The arguments.</param>
22
		public static void Main(string[] args)
23
		{
24
			if (args.Length != 1)
25
			{
26
				Console.WriteLine("Missing path to TFS project collection.");
27
				return;
28
			}
29
30
			var now = DateTime.Now;
31
			var autoResetEvent = new AutoResetEvent(true);
32
			var uri = new Uri(args[0]);
33
			var dashboard = new Dashboard(uri);
34
			var configurationServer = TfsConfigurationServerFactory.GetConfigurationServer(uri, new UICredentialsProvider());
35
			configurationServer.EnsureAuthenticated();
36
37
			// Start the sound board.
38
			var soundBoard = new BackgroundWorker();
39
			soundBoard.DoWork += (o, a) =>
40
			{
41
				while (!a.Cancel)
42
				{
43
					if (Sound.Queue.Count > 0)
44
					{
45
						Sound.Queue.Dequeue()();
46
					}
47
				}
48
			};
49
			soundBoard.RunWorkerAsync();
50
51
			// Start the TFS worker.
52
			var worker = new BackgroundWorker();
53
			worker.DoWork += (o, a) =>
54
			{
55
				now = DateTime.Now;
56
				dashboard.ClearHighlights();
57
				Debug.WriteLine("Worker started at " + DateTime.Now);
58
59
				if (a.Cancel)
60
				{
61
					return;
62
				}
63
64
				// Get all the project collections.
65
				var collections = configurationServer.CatalogNode.QueryChildren(
66
					new[] { CatalogResourceTypes.ProjectCollection },
67
					false,
68
					CatalogQueryOptions.None);
69
70
				foreach (var collection in collections)
71
				{
72
					if (a.Cancel)
73
					{
74
						return;
75
					}
76
77
					// Find the project collection.
78
					var projectCollectionId = new Guid(collection.Resource.Properties["InstanceId"]);
79
					var projectCollection = configurationServer.GetTeamProjectCollection(projectCollectionId);
80
81
					if (a.Cancel)
82
					{
83
						return;
84
					}
85
86
					// Get the projects.
87
					var projects = projectCollection.CatalogNode.QueryChildren(
88
						new[] { CatalogResourceTypes.TeamProject },
89
						false,
90
						CatalogQueryOptions.None);
91
					dashboard.AddProjects(projects.ToArray());
92
93
					if (a.Cancel)
94
					{
95
						return;
96
					}
97
98
					// Get the work history.
99
					var workItemStore = projectCollection.GetService<WorkItemStore>();
100
					var workItemCollection = workItemStore.Query("select * from workitems order by [changed date] asc");
101
					if (dashboard.DateRefresh.HasValue)
102
					{
103
						dashboard.AddWorkItems(workItemCollection.Cast<WorkItem>().Where(x => x.ChangedDate > dashboard.DateRefresh.Value));
104
					}
105
					else
106
					{
107
						dashboard.AddWorkItems(workItemCollection.Cast<WorkItem>());
108
					}
109
110
					if (a.Cancel)
111
					{
112
						return;
113
					}
114
115
					// Get the commit log.
116
					var versionControlServer = projectCollection.GetService<VersionControlServer>();
117
					var commits = versionControlServer.QueryHistory("$/", RecursionType.Full);
118
					if (dashboard.DateRefresh.HasValue)
119
					{
120
						dashboard.AddCommits(commits.Where(x => x.CreationDate > dashboard.DateRefresh.Value));
121
					}
122
					else
123
					{
124
						dashboard.AddCommits(commits);
125
					}
126
				}
127
			};
128
			worker.RunWorkerCompleted += (o, a) =>
129
			{
130
				Debug.WriteLine("Worker completed at " + DateTime.Now);
131
132
				dashboard.DateRefresh = now;
133
				dashboard.Cache();
134
				dashboard.Update();
135
136
				worker.RunWorkerAsync();
137
			};
138
			worker.RunWorkerAsync();
139
140
			var running = true;
141
			while (running)
142
			{
143
				dashboard.Update();
144
145
				// Listen for key events.
146
				if (Console.KeyAvailable)
147
				{
148
					var key = Console.ReadKey(true);
149
						
150
					switch (key.Key)
151
					{
152
						case ConsoleKey.PageDown:
153
							dashboard.SelectedItem += 20;
154
							break;
155
156
						case ConsoleKey.J:
157
						case ConsoleKey.DownArrow:
158
							// Move down.
159
							dashboard.SelectedItem++;
160
							break;
161
162
						case ConsoleKey.PageUp:
163
							dashboard.SelectedItem -= 20;
164
							if (dashboard.SelectedItem < 1)
165
							{
166
								dashboard.SelectedItem = 1;
167
							}
168
							break;
169
170
						case ConsoleKey.UpArrow:
171
						case ConsoleKey.K:
172
							// Move up.
173
							if (dashboard.SelectedItem > 0)
174
							{
175
								dashboard.SelectedItem--;
176
							}
177
							break;
178
179
						case ConsoleKey.Escape:
180
						case ConsoleKey.Q:
181
							// Exit the program.
182
							running = false;
183
							break;
184
185
						default:
186
							Debug.WriteLine("Unhandled key press: " + key.Key);
187
							break;
188
					}
189
				}
190
191
				if (running && autoResetEvent.WaitOne(new TimeSpan(0, 0, 0, 0, 100)))
192
				{
193
					autoResetEvent.Reset();
194
				}
195
			}
196
197
			if (autoResetEvent.WaitOne(new TimeSpan(0, 0, 5)))
198
			{
199
				// Clean-up.
200
				try
201
				{
202
					worker.CancelAsync();
203
				}
204
				catch
205
				{
206
				}
207
208
				try
209
				{
210
					soundBoard.CancelAsync();
211
				}
212
				catch
213
				{
214
				}
215
			}
216
		}
217
	}
218
}

+ 36 - 0
Properties/AssemblyInfo.cs

@ -0,0 +1,36 @@
1
using System.Reflection;
2
using System.Runtime.CompilerServices;
3
using System.Runtime.InteropServices;
4
5
// General Information about an assembly is controlled through the following 
6
// set of attributes. Change these attribute values to modify the information
7
// associated with an assembly.
8
[assembly: AssemblyTitle("pulse")]
9
[assembly: AssemblyDescription("TFS 2013 leadership board to promotoe friendly competition amongst co-workers")]
10
[assembly: AssemblyConfiguration("")]
11
[assembly: AssemblyCompany("Revolving Cow, LLC")]
12
[assembly: AssemblyProduct("pulse")]
13
[assembly: AssemblyCopyright("Copyright Revolving Cow, LLC © 2014")]
14
[assembly: AssemblyTrademark("")]
15
[assembly: AssemblyCulture("")]
16
17
// Setting ComVisible to false makes the types in this assembly not visible 
18
// to COM components.  If you need to access a type in this assembly from 
19
// COM, set the ComVisible attribute to true on that type.
20
[assembly: ComVisible(false)]
21
22
// The following GUID is for the ID of the typelib if this project is exposed to COM
23
[assembly: Guid("887bd478-cfa8-49fe-b84f-ef35f5dc373e")]
24
25
// Version information for an assembly consists of the following four values:
26
//
27
//      Major Version
28
//      Minor Version 
29
//      Build Number
30
//      Revision
31
//
32
// You can specify all the values or you can default the Build and Revision Numbers 
33
// by using the '*' as shown below:
34
// [assembly: AssemblyVersion("1.0.*")]
35
[assembly: AssemblyVersion("1.0.0.0")]
36
[assembly: AssemblyFileVersion("1.0.0.0")]

+ 30 - 0
README

@ -0,0 +1,30 @@
1
pulse
2
=====
3
4
`pulse` is a console application with the sole purpose of promoting friendly competition
5
amongst co-workers. This version only connects to TFS (we tested only version 2013 but
6
we assume it will work with Visual Studio Online) at the moment.
7
8
How to run the program
9
======================
10
11
"""
12
pulse.exe "https://server:443/defaultcollection"
13
"""
14
15
Navigation
16
==========
17
18
To highlight a particular player simply use one of the following commands to highlight
19
the respective line:
20
21
 - Page Up/Down: Move up/down by 20 lines
22
 - Move Up/Down: Either in Vim tradition (J or K) or the arrow keys
23
 - Quit: Either in Vim tradition (Q) or with the Escape key
24
25
Maintenance
26
===========
27
28
After each successful pull of the commit log, work history, and projects the current
29
leaderboard is cached. This cache file is in simple delimited format so feel free to take
30
a look, but please no fudging the numbers! It only hurts morale...

+ 53 - 0
Rewards.cs

@ -0,0 +1,53 @@
1
namespace pulse
2
{
3
	/// <summary>
4
	/// Reward point values.
5
	/// </summary>
6
	public static class Rewards
7
	{
8
		/// <summary>
9
		/// The participation.
10
		/// </summary>
11
		public const double Participation = 0.3;
12
13
		/// <summary>
14
		/// The verbosity.
15
		/// </summary>
16
		public const double Verbosity = 0.8;
17
18
		/// <summary>
19
		/// The collaboration.
20
		/// </summary>
21
		public const double Collaboration = 0.7;
22
23
		/// <summary>
24
		/// The house cleaning.
25
		/// </summary>
26
		public const double HouseCleaning = 0.1;
27
28
		/// <summary>
29
		/// The due diligence.
30
		/// </summary>
31
		public const double DueDiligence = 0.4;
32
33
		/// <summary>
34
		/// The bug report.
35
		/// </summary>
36
		public const double BugReport = 0.7;
37
38
		/// <summary>
39
		/// The created work.
40
		/// </summary>
41
		public const double CreatedWork = 0.02;
42
43
		/// <summary>
44
		/// The finish work.
45
		/// </summary>
46
		public const double FinishWork = 0.5;
47
48
		/// <summary>
49
		/// The taking ownership.
50
		/// </summary>
51
		public const double TakingOwnership = 0.1;
52
	}
53
}

+ 42 - 0
Sound.cs

@ -0,0 +1,42 @@
1
using System;
2
using System.Collections.Generic;
3
using System.Threading;
4
5
namespace pulse
6
{
7
	/// <summary>
8
	/// Sound class.
9
	/// </summary>
10
	public static class Sound
11
	{
12
		/// <summary>
13
		/// The queue
14
		/// </summary>
15
		public static Queue<Action> Queue = new Queue<Action>();
16
17
		/// <summary>
18
		/// Tetrises this instance.
19
		/// </summary>
20
		public static void Tetris()
21
		{
22
			Console.Beep(658, 125); Console.Beep(1320, 500); Console.Beep(990, 250); Console.Beep(1056, 250); Console.Beep(1188, 250); Console.Beep(1320, 125); Console.Beep(1188, 125); Console.Beep(1056, 250); Console.Beep(990, 250); Console.Beep(880, 500); Console.Beep(880, 250); Console.Beep(1056, 250); Console.Beep(1320, 500); Console.Beep(1188, 250); Console.Beep(1056, 250); Console.Beep(990, 750); Console.Beep(1056, 250); Console.Beep(1188, 500); Console.Beep(1320, 500); Console.Beep(1056, 500); Console.Beep(880, 500); Console.Beep(880, 500); Thread.Sleep(250); Console.Beep(1188, 500); Console.Beep(1408, 250); Console.Beep(1760, 500); Console.Beep(1584, 250); Console.Beep(1408, 250); Console.Beep(1320, 750); Console.Beep(1056, 250); Console.Beep(1320, 500); Console.Beep(1188, 250); Console.Beep(1056, 250); Console.Beep(990, 500); Console.Beep(990, 250); Console.Beep(1056, 250); Console.Beep(1188, 500); Console.Beep(1320, 500); Console.Beep(1056, 500); Console.Beep(880, 500); Console.Beep(880, 500); Thread.Sleep(500); Console.Beep(1320, 500); Console.Beep(990, 250); Console.Beep(1056, 250); Console.Beep(1188, 250); Console.Beep(1320, 125); Console.Beep(1188, 125); Console.Beep(1056, 250); Console.Beep(990, 250); Console.Beep(880, 500); Console.Beep(880, 250); Console.Beep(1056, 250); Console.Beep(1320, 500); Console.Beep(1188, 250); Console.Beep(1056, 250); Console.Beep(990, 750); Console.Beep(1056, 250); Console.Beep(1188, 500); Console.Beep(1320, 500); Console.Beep(1056, 500); Console.Beep(880, 500); Console.Beep(880, 500); Thread.Sleep(250); Console.Beep(1188, 500); Console.Beep(1408, 250); Console.Beep(1760, 500); Console.Beep(1584, 250); Console.Beep(1408, 250); Console.Beep(1320, 750); Console.Beep(1056, 250); Console.Beep(1320, 500); Console.Beep(1188, 250); Console.Beep(1056, 250); Console.Beep(990, 500); Console.Beep(990, 250); Console.Beep(1056, 250); Console.Beep(1188, 500); Console.Beep(1320, 500); Console.Beep(1056, 500); Console.Beep(880, 500); Console.Beep(880, 500); Thread.Sleep(500); Console.Beep(660, 1000); Console.Beep(528, 1000); Console.Beep(594, 1000); Console.Beep(495, 1000); Console.Beep(528, 1000); Console.Beep(440, 1000); Console.Beep(419, 1000); Console.Beep(495, 1000); Console.Beep(660, 1000); Console.Beep(528, 1000); Console.Beep(594, 1000); Console.Beep(495, 1000); Console.Beep(528, 500); Console.Beep(660, 500); Console.Beep(880, 1000); Console.Beep(838, 2000); Console.Beep(660, 1000); Console.Beep(528, 1000); Console.Beep(594, 1000); Console.Beep(495, 1000); Console.Beep(528, 1000); Console.Beep(440, 1000); Console.Beep(419, 1000); Console.Beep(495, 1000); Console.Beep(660, 1000); Console.Beep(528, 1000); Console.Beep(594, 1000); Console.Beep(495, 1000); Console.Beep(528, 500); Console.Beep(660, 500); Console.Beep(880, 1000); Console.Beep(838, 2000);
23
		}
24
25
		/// <summary>
26
		/// Marioes this instance.
27
		/// </summary>
28
		public static void Mario()
29
		{
30
			Console.Beep(659, 125); Console.Beep(659, 125); Thread.Sleep(125); Console.Beep(659, 125); Thread.Sleep(167); Console.Beep(523, 125); Console.Beep(659, 125); Thread.Sleep(125); Console.Beep(784, 125); Thread.Sleep(375); Console.Beep(392, 125); Thread.Sleep(375); Console.Beep(523, 125); Thread.Sleep(250); Console.Beep(392, 125); Thread.Sleep(250); Console.Beep(330, 125); Thread.Sleep(250); Console.Beep(440, 125); Thread.Sleep(125); Console.Beep(494, 125); Thread.Sleep(125); Console.Beep(466, 125); Thread.Sleep(42); Console.Beep(440, 125); Thread.Sleep(125); Console.Beep(392, 125); Thread.Sleep(125); Console.Beep(659, 125); Thread.Sleep(125); Console.Beep(784, 125); Thread.Sleep(125); Console.Beep(880, 125); Thread.Sleep(125); Console.Beep(698, 125); Console.Beep(784, 125); Thread.Sleep(125); Console.Beep(659, 125); Thread.Sleep(125); Console.Beep(523, 125); Thread.Sleep(125); Console.Beep(587, 125); Console.Beep(494, 125); Thread.Sleep(125); Console.Beep(523, 125); Thread.Sleep(250); Console.Beep(392, 125); Thread.Sleep(250); Console.Beep(330, 125); Thread.Sleep(250); Console.Beep(440, 125); Thread.Sleep(125); Console.Beep(494, 125); Thread.Sleep(125); Console.Beep(466, 125); Thread.Sleep(42); Console.Beep(440, 125); Thread.Sleep(125); Console.Beep(392, 125); Thread.Sleep(125); Console.Beep(659, 125); Thread.Sleep(125); Console.Beep(784, 125); Thread.Sleep(125); Console.Beep(880, 125); Thread.Sleep(125); Console.Beep(698, 125); Console.Beep(784, 125); Thread.Sleep(125); Console.Beep(659, 125); Thread.Sleep(125); Console.Beep(523, 125); Thread.Sleep(125); Console.Beep(587, 125); Console.Beep(494, 125); Thread.Sleep(375); Console.Beep(784, 125); Console.Beep(740, 125); Console.Beep(698, 125); Thread.Sleep(42); Console.Beep(622, 125); Thread.Sleep(125); Console.Beep(659, 125); Thread.Sleep(167); Console.Beep(415, 125); Console.Beep(440, 125); Console.Beep(523, 125); Thread.Sleep(125); Console.Beep(440, 125); Console.Beep(523, 125); Console.Beep(587, 125); Thread.Sleep(250); Console.Beep(784, 125); Console.Beep(740, 125); Console.Beep(698, 125); Thread.Sleep(42); Console.Beep(622, 125); Thread.Sleep(125); Console.Beep(659, 125); Thread.Sleep(167); Console.Beep(698, 125); Thread.Sleep(125); Console.Beep(698, 125); Console.Beep(698, 125); Thread.Sleep(625); Console.Beep(784, 125); Console.Beep(740, 125); Console.Beep(698, 125); Thread.Sleep(42); Console.Beep(622, 125); Thread.Sleep(125); Console.Beep(659, 125); Thread.Sleep(167); Console.Beep(415, 125); Console.Beep(440, 125); Console.Beep(523, 125); Thread.Sleep(125); Console.Beep(440, 125); Console.Beep(523, 125); Console.Beep(587, 125); Thread.Sleep(250); Console.Beep(622, 125); Thread.Sleep(250); Console.Beep(587, 125); Thread.Sleep(250); Console.Beep(523, 125); Thread.Sleep(1125); Console.Beep(784, 125); Console.Beep(740, 125); Console.Beep(698, 125); Thread.Sleep(42); Console.Beep(622, 125); Thread.Sleep(125); Console.Beep(659, 125); Thread.Sleep(167); Console.Beep(415, 125); Console.Beep(440, 125); Console.Beep(523, 125); Thread.Sleep(125); Console.Beep(440, 125); Console.Beep(523, 125); Console.Beep(587, 125); Thread.Sleep(250); Console.Beep(784, 125); Console.Beep(740, 125); Console.Beep(698, 125); Thread.Sleep(42); Console.Beep(622, 125); Thread.Sleep(125); Console.Beep(659, 125); Thread.Sleep(167); Console.Beep(698, 125); Thread.Sleep(125); Console.Beep(698, 125); Console.Beep(698, 125); Thread.Sleep(625); Console.Beep(784, 125); Console.Beep(740, 125); Console.Beep(698, 125); Thread.Sleep(42); Console.Beep(622, 125); Thread.Sleep(125); Console.Beep(659, 125); Thread.Sleep(167); Console.Beep(415, 125); Console.Beep(440, 125); Console.Beep(523, 125); Thread.Sleep(125); Console.Beep(440, 125); Console.Beep(523, 125); Console.Beep(587, 125); Thread.Sleep(250); Console.Beep(622, 125); Thread.Sleep(250); Console.Beep(587, 125); Thread.Sleep(250); Console.Beep(523, 125);
31
		}
32
33
		/// <summary>
34
		/// Notifies this instance.
35
		/// </summary>
36
		public static void Notify()
37
		{
38
			Console.Beep(658, 125);
39
			Console.Beep(1320, 500);
40
		}
41
	}
42
}

+ 34 - 0
TfsCredentialsProvider.cs

@ -0,0 +1,34 @@
1
using System;
2
using System.Net;
3
using Microsoft.TeamFoundation.Client;
4
5
namespace pulse
6
{
7
	/// <summary>
8
	/// TFS credential provider.
9
	/// </summary>
10
	/// <remarks>Currently not used but may want to.</remarks>
11
	public class TfsCredentialProvider : ICredentialsProvider
12
	{
13
		/// <summary>
14
		/// Gets the credentials.
15
		/// </summary>
16
		/// <param name="uri">The URI.</param>
17
		/// <param name="failedCredentials">The failed credentials.</param>
18
		/// <returns></returns>
19
		public ICredentials GetCredentials(Uri uri, ICredentials failedCredentials)
20
		{
21
			return new NetworkCredential("UserName", "Password", "Domain");
22
		}
23
24
		/// <summary>
25
		/// Notifies the credentials authenticated.
26
		/// </summary>
27
		/// <param name="uri">The URI.</param>
28
		/// <exception cref="System.NotImplementedException"></exception>
29
		public void NotifyCredentialsAuthenticated(Uri uri)
30
		{
31
			throw new NotImplementedException();
32
		}
33
	}
34
}

+ 63 - 0
pulse.csproj

@ -0,0 +1,63 @@
1
<?xml version="1.0" encoding="utf-8"?>
2
<Project ToolsVersion="12.0" DefaultTargets="Build" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
3
  <Import Project="$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props" Condition="Exists('$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props')" />
4
  <PropertyGroup>
5
    <Configuration Condition=" '$(Configuration)' == '' ">Debug</Configuration>
6
    <Platform Condition=" '$(Platform)' == '' ">AnyCPU</Platform>
7
    <ProjectGuid>{719EF15E-885F-4788-BDDC-F7384C13BADA}</ProjectGuid>
8
    <OutputType>Exe</OutputType>
9
    <AppDesignerFolder>Properties</AppDesignerFolder>
10
    <RootNamespace>pulse</RootNamespace>
11
    <AssemblyName>pulse</AssemblyName>
12
    <TargetFrameworkVersion>v4.5</TargetFrameworkVersion>
13
    <FileAlignment>512</FileAlignment>
14
  </PropertyGroup>
15
  <PropertyGroup Condition=" '$(Configuration)|$(Platform)' == 'Debug|AnyCPU' ">
16
    <PlatformTarget>AnyCPU</PlatformTarget>
17
    <DebugSymbols>true</DebugSymbols>
18
    <DebugType>full</DebugType>
19
    <Optimize>false</Optimize>
20
    <OutputPath>bin\Debug\</OutputPath>
21
    <DefineConstants>DEBUG;TRACE</DefineConstants>
22
    <ErrorReport>prompt</ErrorReport>
23
    <WarningLevel>4</WarningLevel>
24
  </PropertyGroup>
25
  <PropertyGroup Condition=" '$(Configuration)|$(Platform)' == 'Release|AnyCPU' ">
26
    <PlatformTarget>AnyCPU</PlatformTarget>
27
    <DebugType>pdbonly</DebugType>
28
    <Optimize>true</Optimize>
29
    <OutputPath>bin\Release\</OutputPath>
30
    <DefineConstants>TRACE</DefineConstants>
31
    <ErrorReport>prompt</ErrorReport>
32
    <WarningLevel>4</WarningLevel>
33
  </PropertyGroup>
34
  <ItemGroup>
35
    <Reference Include="Microsoft.TeamFoundation.Client, Version=12.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL" />
36
    <Reference Include="Microsoft.TeamFoundation.Common, Version=12.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL" />
37
    <Reference Include="Microsoft.TeamFoundation.VersionControl.Client, Version=12.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL" />
38
    <Reference Include="Microsoft.TeamFoundation.WorkItemTracking.Client, Version=12.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL" />
39
    <Reference Include="System" />
40
    <Reference Include="System.Core" />
41
    <Reference Include="Microsoft.CSharp" />
42
  </ItemGroup>
43
  <ItemGroup>
44
    <Compile Include="Dashboard.cs" />
45
    <Compile Include="Penalties.cs" />
46
    <Compile Include="Program.cs" />
47
    <Compile Include="Properties\AssemblyInfo.cs" />
48
    <Compile Include="Rewards.cs" />
49
    <Compile Include="Sound.cs" />
50
    <Compile Include="TfsCredentialsProvider.cs" />
51
  </ItemGroup>
52
  <ItemGroup>
53
    <None Include="App.config" />
54
  </ItemGroup>
55
  <Import Project="$(MSBuildToolsPath)\Microsoft.CSharp.targets" />
56
  <!-- To modify your build process, add your task inside one of the targets below and uncomment it. 
57
       Other similar extension points exist, see Microsoft.Common.targets.
58
  <Target Name="BeforeBuild">
59
  </Target>
60
  <Target Name="AfterBuild">
61
  </Target>
62
  -->
63
</Project>