Skip to content

Commit

Permalink
Export all users
Browse files Browse the repository at this point in the history
  • Loading branch information
satano committed Nov 21, 2024
1 parent 9135fee commit 4c98820
Show file tree
Hide file tree
Showing 5 changed files with 83 additions and 32 deletions.
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.IO;
using System.Linq;
using Microsoft.Extensions.Logging;
using Microsoft.Extensions.Options;
Expand Down Expand Up @@ -42,30 +43,26 @@ protected override void InternalExecute()
{
Stopwatch stopwatch = Stopwatch.StartNew();

if (string.IsNullOrEmpty(CommonTools.UserMapping.Options.UserMappingFile))
{
Log.LogError("UserMappingFile is not set");
throw new ArgumentNullException("UserMappingFile must be set on the TfsUserMappingToolOptions in CommonEnrichersConfig.");
}
CheckOptions();

List<IdentityMapData> usersToMap = new List<IdentityMapData>();
IdentityMapResult data;
if (Options.OnlyListUsersInWorkItems)
{
Log.LogInformation("OnlyListUsersInWorkItems is true, only users in work items will be listed");
List<WorkItemData> sourceWorkItems = Source.WorkItems.GetWorkItems(Options.WIQLQuery);
Log.LogInformation("Processed {0} work items from Source", sourceWorkItems.Count);

usersToMap = CommonTools.UserMapping.GetUsersInSourceMappedToTargetForWorkItems(this, sourceWorkItems);
Log.LogInformation("Found {usersToMap} total mapped", usersToMap.Count);
data = CommonTools.UserMapping.GetUsersInSourceMappedToTargetForWorkItems(this, sourceWorkItems);
Log.LogInformation("Found {usersToMap} total mapped", data.IdentityMap.Count);
}
else
{
Log.LogInformation("OnlyListUsersInWorkItems is false, all users will be listed");
usersToMap = CommonTools.UserMapping.GetUsersInSourceMappedToTarget(this);
Log.LogInformation("Found {usersToMap} total mapped", usersToMap.Count);
data = CommonTools.UserMapping.GetUsersInSourceMappedToTarget(this);
Log.LogInformation("Found {usersToMap} total mapped", data.IdentityMap.Count);
}

usersToMap = usersToMap.Where(x => x.Source.DisplayName != x.Target?.DisplayName).ToList();
List<IdentityMapData> usersToMap = data.IdentityMap.Where(x => x.Source.DisplayName != x.Target?.DisplayName).ToList();
Log.LogInformation("Filtered to {usersToMap} total viable mappings", usersToMap.Count);
Dictionary<string, string> usermappings = [];
foreach (IdentityMapData userMapping in usersToMap)
Expand All @@ -74,11 +71,40 @@ protected override void InternalExecute()
// it would throw with duplicate key. This way we just overwrite the value – last item in source wins.
usermappings[userMapping.Source.DisplayName] = userMapping.Target?.DisplayName;
}
System.IO.File.WriteAllText(CommonTools.UserMapping.Options.UserMappingFile, JsonConvert.SerializeObject(usermappings, Formatting.Indented));
Log.LogInformation("Writen to: {LocalExportJsonFile}", CommonTools.UserMapping.Options.UserMappingFile);
File.WriteAllText(CommonTools.UserMapping.Options.UserMappingFile, JsonConvert.SerializeObject(usermappings, Formatting.Indented));
Log.LogInformation("User mappings writen to: {LocalExportJsonFile}", CommonTools.UserMapping.Options.UserMappingFile);
if (Options.ExportAllUsers)
{
ExportAllUsers(data);
}

stopwatch.Stop();
Log.LogInformation("DONE in {Elapsed} seconds", stopwatch.Elapsed);
}

private void ExportAllUsers(IdentityMapResult data)
{
var allUsers = new
{
data.SourceUsers,
data.TargetUsers
};
File.WriteAllText(Options.UserExportFile, JsonConvert.SerializeObject(allUsers, Formatting.Indented));
Log.LogInformation("All user writen to: {exportFile}", Options.UserExportFile);
}

private void CheckOptions()
{
if (string.IsNullOrEmpty(CommonTools.UserMapping.Options.UserMappingFile))
{
Log.LogError("UserMappingFile is not set");
throw new ArgumentNullException("UserMappingFile must be set on the TfsUserMappingToolOptions in CommonTools.");
}
if (Options.ExportAllUsers && string.IsNullOrEmpty(Options.UserExportFile))
{
Log.LogError($"Flag ExportAllUsers is set but export file UserExportFile is not set.");
throw new ArgumentNullException("UserExportFile must be set on the TfsExportUsersForMappingProcessorOptions in Processors.");
}
}
}
}
Original file line number Diff line number Diff line change
@@ -1,7 +1,4 @@
using System.Collections.Generic;
using MigrationTools._EngineV1.Configuration;
using MigrationTools.Enrichers;
using MigrationTools.Processors.Infrastructure;
using MigrationTools.Processors.Infrastructure;

namespace MigrationTools.Processors
{
Expand All @@ -16,5 +13,17 @@ public class TfsExportUsersForMappingProcessorOptions : ProcessorOptions
/// <default>true</default>
public bool OnlyListUsersInWorkItems { get; set; } = true;

/// <summary>
/// Set to <see langword="true"/>, if you want to export all users in source and target server.
/// The lists of user can be useful, if you need tu manually edit mapping file.
/// Users will be exported to file set in <see cref="UserExportFile"/>.
/// </summary>
public bool ExportAllUsers { get; set; }

/// <summary>
/// Path to export file where all source and target servers' users will be exported.
/// Users are exported only if <see cref="ExportAllUsers"/> is set to <see langword="true"/>.
/// </summary>
public string UserExportFile { get; set; }
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -227,15 +227,15 @@ protected override void InternalExecute()

private void ValidateAllUsersExistOrAreMapped(List<WorkItemData> sourceWorkItems)
{

contextLog.Information("Validating::Check that all users in the source exist in the target or are mapped!");
List<IdentityMapData> usersToMap = new List<IdentityMapData>();
usersToMap = CommonTools.UserMapping.GetUsersInSourceMappedToTargetForWorkItems(this, sourceWorkItems);
if (usersToMap != null && usersToMap?.Count > 0)
IdentityMapResult usersToMap = CommonTools.UserMapping.GetUsersInSourceMappedToTargetForWorkItems(this, sourceWorkItems);
if (usersToMap.IdentityMap != null && usersToMap.IdentityMap.Count > 0)
{
Log.LogWarning("Validating Failed! There are {usersToMap} users that exist in the source that do not exist in the target. This will not cause any errors, but may result in disconnected users that could have been mapped. Use the ExportUsersForMapping processor to create a list of mappable users. Then Import using ", usersToMap.Count);
Log.LogWarning("Validating Failed! There are {usersToMap} users that exist in the source that do not exist "
+ "in the target. This will not cause any errors, but may result in disconnected users that could have "
+ "been mapped. Use the ExportUsersForMapping processor to create a list of mappable users.",
usersToMap.IdentityMap.Count);
}

}

//private void ValidateAllNodesExistOrAreMapped(List<WorkItemData> sourceWorkItems)
Expand Down
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
using System;
using System;
using System.Collections.Generic;
using System.Linq;
using Microsoft.Extensions.Logging;
Expand Down Expand Up @@ -121,11 +121,12 @@ private List<IdentityItemData> GetUsersListFromServer(IGroupSecurityService gss)
Log.LogWarning("TfsUserMappingTool::GetUsersListFromServer::[user:{user}] Failed With {Exception}", sid, ex.Message);
}
}
foundUsers.Sort((x, y) => x.AccountName.CompareTo(y.AccountName));
Log.LogInformation("TfsUserMappingTool::GetUsersListFromServer {count} user identities are applicable for mapping", foundUsers.Count);
return foundUsers;
}

public List<IdentityMapData> GetUsersInSourceMappedToTarget(TfsProcessor processor)
public IdentityMapResult GetUsersInSourceMappedToTarget(TfsProcessor processor)
{
Log.LogDebug("TfsUserMappingTool::GetUsersInSourceMappedToTarget");
if (Options.Enabled)
Expand Down Expand Up @@ -162,25 +163,31 @@ public List<IdentityMapData> GetUsersInSourceMappedToTarget(TfsProcessor process
targetUser ??= targetUsers.SingleOrDefault(x => x.DisplayName == sourceUser.DisplayName);
identityMap.Add(new IdentityMapData { Source = sourceUser, Target = targetUser });
}
return identityMap;
return new()
{
IdentityMap = identityMap,
SourceUsers = sourceUsers,
TargetUsers = targetUsers
};
}
else
{
Log.LogWarning("TfsUserMappingTool is disabled in settings. You may have users in the source that are not mapped to the target. ");
return [];
return new();
}
}

public List<IdentityMapData> GetUsersInSourceMappedToTargetForWorkItems(TfsProcessor processor, List<WorkItemData> sourceWorkItems)
public IdentityMapResult GetUsersInSourceMappedToTargetForWorkItems(TfsProcessor processor, List<WorkItemData> sourceWorkItems)
{
if (Options.Enabled)
{
Dictionary<string, string> result = new Dictionary<string, string>();
HashSet<string> workItemUsers = GetUsersFromWorkItems(sourceWorkItems, Options.IdentityFieldsToCheck);
Log.LogDebug($"TfsUserMappingTool::GetUsersInSourceMappedToTargetForWorkItems [workItemUsers|{workItemUsers.Count}]");
List<IdentityMapData> mappedUsers = GetUsersInSourceMappedToTarget(processor);
Log.LogDebug($"TfsUserMappingTool::GetUsersInSourceMappedToTargetForWorkItems [mappedUsers|{mappedUsers.Count}]");
return mappedUsers.Where(x => workItemUsers.Contains(x.Source.DisplayName)).ToList();
IdentityMapResult mappedUsers = GetUsersInSourceMappedToTarget(processor);
Log.LogDebug($"TfsUserMappingTool::GetUsersInSourceMappedToTargetForWorkItems [mappedUsers|{mappedUsers.IdentityMap.Count}]");
mappedUsers.IdentityMap = mappedUsers.IdentityMap.Where(x => workItemUsers.Contains(x.Source.DisplayName)).ToList();
return mappedUsers;
}
else
{
Expand Down
11 changes: 10 additions & 1 deletion src/MigrationTools/DataContracts/IdentityItemData.cs
Original file line number Diff line number Diff line change
@@ -1,4 +1,6 @@
namespace MigrationTools.DataContracts
using System.Collections.Generic;

namespace MigrationTools.DataContracts
{
public class IdentityItemData
{
Expand All @@ -14,4 +16,11 @@ public class IdentityMapData
public IdentityItemData Source { get; set; }
public IdentityItemData Target { get; set; }
}

public class IdentityMapResult
{
public List<IdentityMapData> IdentityMap { get; set; } = [];
public List<IdentityItemData> SourceUsers { get; set; } = [];
public List<IdentityItemData> TargetUsers { get; set; } = [];
}
}

0 comments on commit 4c98820

Please sign in to comment.