|
| 1 | +using Kusto.Language; |
| 2 | +using KustoSchemaTools.Model; |
| 3 | +using KustoSchemaTools.Parser; |
| 4 | +using System.Text; |
| 5 | + |
| 6 | +namespace KustoSchemaTools.Changes |
| 7 | +{ |
| 8 | + /// <summary> |
| 9 | + /// Permission diff for follower databases. Uses follower-specific commands so |
| 10 | + /// we don't attempt to write directly to the read-only follower database. |
| 11 | + /// </summary> |
| 12 | + public class FollowerPermissionChange : BaseChange<List<AADObject>> |
| 13 | + { |
| 14 | + public FollowerPermissionChange(string db, string entity, List<AADObject> from, List<AADObject> to, string? leaderName, string? currentLeaderName) |
| 15 | + : base("FollowerPermissions", entity, from ?? new List<AADObject>(), to) |
| 16 | + { |
| 17 | + Db = db; |
| 18 | + LeaderName = leaderName; |
| 19 | + CurrentLeaderName = currentLeaderName; |
| 20 | + Init(); |
| 21 | + } |
| 22 | + |
| 23 | + public string Db { get; } |
| 24 | + public string? LeaderName { get; } |
| 25 | + public string? CurrentLeaderName { get; } |
| 26 | + |
| 27 | + private string BuildCommand(List<AADObject> principals, string? leaderName) |
| 28 | + { |
| 29 | + // Order-insensitive: sort by Id so equivalent sets don't emit churn |
| 30 | + var ids = principals |
| 31 | + .OrderBy(p => p.Id, StringComparer.OrdinalIgnoreCase) |
| 32 | + .Select(a => "\"" + a.Id + "\"" |
| 33 | + ); |
| 34 | + |
| 35 | + // Kusto control commands expect the literal keyword 'none' when no principals |
| 36 | + // are supplied. Wrapping none in parentheses makes it a principal named "none" |
| 37 | + // and the command is rejected, so only use parentheses for non-empty sets. |
| 38 | + var idsFragment = ids.Any() ? $"({string.Join(",", ids)})" : "none"; |
| 39 | + |
| 40 | + // Leader name is optional; when provided we append it the same way the |
| 41 | + // portal does (e.g. '.add follower database DB viewers (...) "leader"'). |
| 42 | + var leaderSuffix = string.IsNullOrWhiteSpace(leaderName) ? string.Empty : $" '{leaderName}'"; |
| 43 | + |
| 44 | + return $".set follower database {Db.BracketIfIdentifier()} {Entity.ToLower()} {idsFragment}{leaderSuffix}"; |
| 45 | + } |
| 46 | + |
| 47 | + private void Init() |
| 48 | + { |
| 49 | + var targetCmd = BuildCommand(To, LeaderName); |
| 50 | + var currentCmd = BuildCommand(From, CurrentLeaderName); |
| 51 | + |
| 52 | + if (!string.Equals(targetCmd, currentCmd, StringComparison.Ordinal)) |
| 53 | + { |
| 54 | + var script = new DatabaseScript { Text = targetCmd, Order = 0 }; |
| 55 | + var container = new DatabaseScriptContainer(script, "FollowerPermissionChange"); |
| 56 | + var code = KustoCode.Parse(script.Text); |
| 57 | + container.IsValid = !code.GetDiagnostics().Any(); |
| 58 | + Scripts.Add(container); |
| 59 | + } |
| 60 | + |
| 61 | + var added = To.Where(itm => From.All(t => t.Id != itm.Id)).ToList(); |
| 62 | + var removed = From.Where(itm => To.All(t => t.Id != itm.Id)).ToList(); |
| 63 | + var changed = From.Join(To, f => f.Id, t => t.Id, (f, t) => new { f, t }) |
| 64 | + .Where(x => x.f.Name != x.t.Name) |
| 65 | + .ToList(); |
| 66 | + |
| 67 | + var sb = new StringBuilder(); |
| 68 | + sb.AppendLine($"## {Entity} (Follower)"); |
| 69 | + sb.AppendLine(); |
| 70 | + sb.AppendLine("<table>"); |
| 71 | + sb.AppendLine("<tr></tr>"); |
| 72 | + |
| 73 | + if (added.Any()) |
| 74 | + { |
| 75 | + sb.AppendLine("<tr><td colspan=\"2\">Added:</td><td colspan=\"10\">" + string.Join("<br>", added.Select(t => $"{t.Name} ({t.Id})")) + "</td></tr>"); |
| 76 | + } |
| 77 | + if (removed.Any()) |
| 78 | + { |
| 79 | + sb.AppendLine("<tr><td colspan=\"2\">Removed:</td><td colspan=\"10\">" + string.Join("<br>", removed.Select(t => $"{t.Name} ({t.Id})")) + "</td></tr>"); |
| 80 | + } |
| 81 | + if (changed.Any()) |
| 82 | + { |
| 83 | + Scripts.Add(new DatabaseScriptContainer("FollowerPermissionRenamed", -1, "// No Database Change")); |
| 84 | + sb.AppendLine("<tr><td colspan=\"2\">Changed:</td><td colspan=\"10\">" + string.Join("<br>", changed.Select(t => $"{t.f.Name} => {t.t.Name} ({t.t.Id})")) + "</td></tr>"); |
| 85 | + } |
| 86 | + |
| 87 | + var logo = Scripts.Any() && Scripts.First().IsValid == false ? ":red_circle:" : ":green_circle:"; |
| 88 | + var displayCmd = Scripts.FirstOrDefault()?.Script?.Text ?? currentCmd; |
| 89 | + sb.AppendLine($"<tr><td colspan=\"2\">{logo}</td><td colspan=\"10\"><pre lang=\"kql\">{displayCmd.PrettifyKql()}</pre></td></tr>"); |
| 90 | + sb.AppendLine("</table>"); |
| 91 | + |
| 92 | + Markdown = sb.ToString(); |
| 93 | + } |
| 94 | + } |
| 95 | +} |
0 commit comments