The last couple of weeks I have been bussy with a lot of code reviews at our Professional Development Center at info support. We are in the finalizing stage of our next release of our software factory Endeavour. During this release colleagues made several changes to an existing code base for bug fixes and additional features. As you might know by now we use Team Foundation Server since August last year for managing our sources and work items. So for my code review work, I wanted a list of C# files that have changed since the last release. (This matches the date we started using TFS, so I can just query Version Control for all changes since we started using TFS)
So I started of with using tf.exe to get me the list I wanted. While tf.exe can deliver you a lot of information, this specific question: Which C# files got changed since the day we added all sources to our repository, does not work completely.
I used queries like :tf history “c:mylocalworkspaceFolder” /recursive /format:detailed /no prompt /version:D08/01/2005
This results in a pretty lengthy list, with all information on all changes on all files made since the 1st of august 2005.
So you can get a list, but this list contains all information on all change sets in all detail, and that was just too much. I could have used it and then use Excel to tweak the list to my needs, but hey I am a programmer and know I can get exactly what I need using the object model of version control. (Excel is just way to complex for meJ)
Just to show you how powerful the object model is, you can see the code below on what is required to get a list of C# files that got modified since a specific version.
using System;
using System.Collections.Generic;
using Microsoft.TeamFoundation.Client;
using Microsoft.TeamFoundation.VersionControl.Client;
namespace GetReviewList
{
class Program
{
static void Main(string[] args)
{
if ((args.Length < 2) || (args.Length > 3))
{
Console.WriteLine(“expected input args: [serverUri]”);
Console.WriteLine(“Sample for server TFSB3 local working folder c:\temp and all changes of C# files after date 12 april 2005”);
Console.WriteLine(“GetReviewList “c:\temp\*.cs” “D04/12/2005” “http://TFSB3:8080″”);
Console.WriteLine(“for alternative version specs, see tfs documentation on version specs for Changesets, Labels, etc.”);
return;
}
string localPath = args[0];
string versionFromString = args[1];
TeamFoundationServer tfs = null;
if (args.Length == 3)
{
tfs = TeamFoundationServerFactory.GetServer( args[2]);
}
else
{
// asume single unique server
tfs = TeamFoundationServerFactory.GetUniqueServer();
}
List<string> changedFiles = new List<string>();
VersionControlServer vcs = (VersionControlServer)tfs.GetService(typeof(VersionControlServer));
try
{
foreach (Changeset item in vcs.QueryHistory(localPath,VersionSpec.ParseSingleSpec(versionFromString,null), 0,RecursionType.Full, null,VersionSpec.ParseSingleSpec(versionFromString, null), null, Int32.MaxValue, true, false))
{
// now get the server items involved
DateTime checkInDate = item.CreationDate;
string user = item.Committer;
foreach (Change changedItem in item.Changes)
{
//filter all other files then c# files, those need
//to be reviewed
//[
// update: removed check on .cs files, based on the feedback of James. When using a file mask in the
// itempec you can query for any type of file. e.g. itemspec: c:temp*.cs would result in all c# files
// Thank you James
//]
string filename = changedItem.Item.ServerItem.Substring(changedItem.Item.ServerItem.LastIndexOf(‘/’) + 1);
filename = (string.Format(“{0}”, filename));
//if you need more info on the changes per file you can also use the string below that shows
//not only the files changed, but also who, when and what was done at checkin.
//filename = (string.Format(“[{0}]t[{1}]t[{2}]t {3}”, user,
// checkInDate.ToString(“dd-MM-yyyy”),changedItem.ChangeType.ToString(), filename));
if (!changedFiles.Contains(filename))
{
changedFiles.Add(filename);
}
}
}
//now print the list of items
foreach (string file in changedFiles)
{
Console.WriteLine(file);
}
}
catch (Exception e)
{Console.WriteLine(e.Message);}
}
}
}
If you would use the program to get your review list, notice you can use also different specifications for the history by specifying not a date but e.g. a change set from which moment on you want to know the changes. The way you would do this is exactly the same as on the tf.exe command line. There is the notion of a version specification that can have the following formats:
Date/time : D10/20/2005
[Update: you can find great info on the date formats you can use at James Mannings blog at the following location]
Changeset number: C1256
Label : Lmylabel
Latest version : T
Workspace: WWorkspaceName[;WorkspaceOwner]
One comment
Good example!
Couple of quick notes:
– since we do "path and pattern", when you’re doing recursive, you could do c:folder*.cs or $/*.cs to get all the .cs files under a given dir
– one alternative approach would be to properties *.cs and check the changeset or checked-in date – if later than cut-off, it’s been changed
– another potential approach would be to diff *.cs recursively /type:binary between that date and tip to see if each file has changed
– this would have secondary effect of ignoring files where the contents hadn’t actually changed (rename/deleted/undelete/etc. types of changes only)
James Manning