I have updated this code to run as a recurring ITeamFoundationJobExtension .
The first file creates the job in TFS to run every 24 hours, and runs it once right now.
The second file is the updated code that is now a TFS Agent Job, this required a few changes to the code to run against the server api instead of the client one. They're fairly similar but the client api tends to be a bit more friendly.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
using Microsoft.TeamFoundation.Framework.Server; | |
using System; | |
using System.Collections.Generic; | |
using System.DirectoryServices.AccountManagement; | |
using System.Linq; | |
using System.Text; | |
using Microsoft.TeamFoundation.Framework.Common; | |
using System.Threading.Tasks; | |
namespace Enlighten.Tfs.ActiveDirectoryImageSync | |
{ | |
public enum ImageFormat | |
{ | |
bmp, | |
jpeg, | |
gif, | |
tiff, | |
png, | |
unknown | |
} | |
public class SyncImagesJob : ITeamFoundationJobExtension | |
{ | |
public TeamFoundationJobExecutionResult Run(TeamFoundationRequestContext requestContext, Microsoft.TeamFoundation.Framework.Server.TeamFoundationJobDefinition jobDefinition, DateTime queueTime, out string resultMessage) | |
{ | |
resultMessage = ""; | |
try | |
{ | |
var service = requestContext.GetService<TeamFoundationIdentityService>(); | |
var identities = service.ReadFilteredIdentities(requestContext, "Microsoft.TeamFoundation.Identity.DisplayName CONTAINS '' AND Microsoft.TeamFoundation.Identity.Type == 'User'", 5000, null, true, MembershipQuery.None); | |
if (identities != null) | |
{ | |
foreach (var identity in identities.Items) | |
{ | |
var user = UserPrincipal.FindByIdentity(new PrincipalContext(ContextType.Domain), identity.UniqueName); | |
if (user == null) continue; | |
var de = new System.DirectoryServices.DirectoryEntry("LDAP://" + user.DistinguishedName); | |
var thumbNail = de.Properties["thumbnailPhoto"].Value as byte[]; | |
if (thumbNail == null) | |
continue; | |
var imageFormat = GetImageFormat(thumbNail); | |
var propertyUpdates = new List<PropertyValue>() { | |
new PropertyValue("Microsoft.TeamFoundation.Identity.Image.Data", thumbNail), | |
new PropertyValue("Microsoft.TeamFoundation.Identity.Image.Type", "image/" + imageFormat.ToString()), | |
new PropertyValue("Microsoft.TeamFoundation.Identity.Image.Id", Guid.NewGuid().ToByteArray()), | |
new PropertyValue("Microsoft.TeamFoundation.Identity.CandidateImage.Data", null), | |
new PropertyValue("Microsoft.TeamFoundation.Identity.CandidateImage.UploadDate", null), | |
}; | |
service.UpdateExtendedProperties(requestContext, identity.Descriptor, propertyUpdates); | |
service.RefreshIdentity(requestContext, identity.Descriptor); | |
} | |
} | |
} | |
catch (Exception e) | |
{ | |
resultMessage = e.Message + e.StackTrace; | |
return TeamFoundationJobExecutionResult.Failed; | |
} | |
return TeamFoundationJobExecutionResult.Succeeded; | |
} | |
//from http://stackoverflow.com/questions/1397512/find-image-format-using-bitmap-object-in-c-sharp | |
private static ImageFormat GetImageFormat(byte[] bytes) | |
{ | |
// see http://www.mikekunz.com/image_file_header.html | |
var bmp = Encoding.ASCII.GetBytes("BM"); // BMP | |
var gif = Encoding.ASCII.GetBytes("GIF"); // GIF | |
var png = new byte[] { 137, 80, 78, 71 }; // PNG | |
var tiff = new byte[] { 73, 73, 42 }; // TIFF | |
var tiff2 = new byte[] { 77, 77, 42 }; // TIFF | |
var jpeg = new byte[] { 255, 216, 255, 224 }; // jpeg | |
var jpeg2 = new byte[] { 255, 216, 255, 225 }; // jpeg canon | |
if (bmp.SequenceEqual(bytes.Take(bmp.Length))) | |
return ImageFormat.bmp; | |
if (gif.SequenceEqual(bytes.Take(gif.Length))) | |
return ImageFormat.gif; | |
if (png.SequenceEqual(bytes.Take(png.Length))) | |
return ImageFormat.png; | |
if (tiff.SequenceEqual(bytes.Take(tiff.Length))) | |
return ImageFormat.tiff; | |
if (tiff2.SequenceEqual(bytes.Take(tiff2.Length))) | |
return ImageFormat.tiff; | |
if (jpeg.SequenceEqual(bytes.Take(jpeg.Length))) | |
return ImageFormat.jpeg; | |
if (jpeg2.SequenceEqual(bytes.Take(jpeg2.Length))) | |
return ImageFormat.jpeg; | |
return ImageFormat.unknown; | |
} | |
} | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
using Microsoft.TeamFoundation.Client; | |
using Microsoft.TeamFoundation.Framework.Client; | |
using Microsoft.TeamFoundation.Framework.Common; | |
using System; | |
using System.Collections.Generic; | |
using System.Linq; | |
using System.Text; | |
using System.Threading.Tasks; | |
using System.Xml; | |
namespace Enlighten.Tfs.ConsoleApp | |
{ | |
public class ConsoleApplication | |
{ | |
public static void Main() | |
{ | |
Uri tfsCollectionUri = new Uri(String.Format("http://urltotfs")); | |
var tfs = TfsTeamProjectCollectionFactory.GetTeamProjectCollection(tfsCollectionUri); | |
var service = tfs.GetService<ITeamFoundationJobService>(); | |
var definition = new TeamFoundationJobDefinition( | |
new Guid("fa60c04e-c996-413e-8151-15933f5a2bac"), | |
"AD Image Sync Job", | |
"Enlighten.Tfs.ActiveDirectoryImageSync.SyncImagesJob", | |
null, | |
TeamFoundationJobEnabledState.Enabled); | |
var schedule = new TeamFoundationJobSchedule(DateTime.Now, 24 * 60 * 60); | |
definition.Schedule.Add(schedule); | |
service.UpdateJob(definition); | |
service.QueueJobNow(definition, false); | |
Console.ReadLine(); | |
} | |
} | |
} |
Copy the compiled job to c:\Program Files\Microsoft Team Foundation Server 11.0\Application Tier\TFSJobAgent\plugins and restart the agent service. Then run the console application to register the job. Finally check the state of the job by looking in the tfs database
select * from tfs_configuration.dbo.tbl_JobHistory where jobid = 'fa60c04e-c996-413e-8151-15933f5a2bac'