-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathPlanner.cs
422 lines (370 loc) · 16.8 KB
/
Planner.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Net;
using System.Net.Http;
using System.Net.Http.Headers;
using System.Runtime.InteropServices;
using System.Text;
using System.Threading;
using System.Threading.Tasks;
using Microsoft.IdentityModel.Clients.ActiveDirectory;
using PlannerExAndImport.JSON;
namespace PlannerExAndImport
{
// connects to the MS Graph API (https://graph.microsoft.com) and ex- and imports groups, plans, tasks, details etc.
public class Planner
{
// URLs and settings for the Graph connection
private const string GRAPH_ENDPOINT = "https://graph.microsoft.com";
private const string PLANNER_SUB = "/v1.0/planner/";
private const string GROUPS_SUB = "/v1.0/groups/";
private const string USERS_SUB = "/v1.0/users/";
private const string RESOURCE_ID = GRAPH_ENDPOINT;
public static string CLIENT_ID = "";
public static string TENANT = "";
private static Dictionary<string, string> users = new Dictionary<string, string>();
// export a plan and optionally output it as json
public static Plan[] Export(bool output = true, bool allowMultiSelect = false, bool retrieveTaskDetail = true)
{
Plan[] plans = SelectPlan(allowMultiSelect);
if (!allowMultiSelect && plans.Length > 1)
{
Console.WriteLine("You can only select 1 plan in this case");
return null;
}
using (var httpClient = PreparePlannerClient())
{
foreach (Plan plan in plans)
{
// get all buckets, tasks and task details
var buckets = GraphResponse<BucketResponse>.Get("plans/" + plan.Id + "/buckets", httpClient).Result.Buckets;
var tasks = GraphResponse<TaskResponse>.Get("plans/" + plan.Id + "/tasks", httpClient).Result.Tasks;
if (retrieveTaskDetail)
foreach (var task in tasks)
task.TaskDetail = GraphResponse<TaskDetailResponse>.Get("tasks/" + task.Id + "/details", httpClient).Result;
// put tasks in buckets so that the plan object has all data hierarchically
foreach (var bucket in buckets)
{
bucket.Tasks = tasks.Where(t => t.BucketId == bucket.Id).ToArray();
}
plan.Buckets = buckets;
}
}
if (output)
{
foreach (Plan plan in plans)
Console.WriteLine(Serialize.ToJson(plan));
}
return plans;
}
internal static void DuplicateBucket()
{
Console.WriteLine("Step 1: Select the bucket to duplicate");
Bucket bucket = SelectBucket();
Console.WriteLine("Step 2: Enter the name for the new bucket");
string newBucketName = Program.GetInput("Which name do you want to use: ");
bucket.Name = newBucketName;
using (var httpClient = PreparePlannerClient())
{
CreateBucket(bucket.PlanId, bucket, true, httpClient);
}
}
// export a plan and import everything into a new plan
public static void Import()
{
Console.WriteLine("Step 1: Select the plan to export");
Plan exportedPlan = Export(false).FirstOrDefault();
Console.WriteLine("Step 2: Select the plan in which you want to import");
Plan targetPlan = SelectPlan(false).FirstOrDefault();
bool addAssignments = Program.GetInput("Do you want to import the assignments (y/n)? This might send email notifications to the assignees. ") == "y";
using (var httpClient = PreparePlannerClient())
{
// buckets and tasks are always added at the beginning, therefore reversing the order when importing, otherwise e.g. the
// last exported bucket would become the first bucket in the imported plan
exportedPlan.Buckets = exportedPlan.Buckets.Reverse().ToArray();
// create buckets and tasks and then set details for the created tasks (can't be done in one step)
foreach (Bucket bucket in exportedPlan.Buckets)
{
CreateBucket(targetPlan.Id, bucket, addAssignments, httpClient);
}
}
Console.WriteLine("Import is done");
}
private static void CreateBucket(string targetPlanId, Bucket bucket, bool addAssignments, HttpClient httpClient)
{
bucket.PlanId = targetPlanId;
// reset all order hints as the exported values don't work
bucket.OrderHint = " !";
var newBucket = GraphResponse<Bucket>.Post("buckets", httpClient, bucket).Result;
// if we are too quick the created bucket is not available yet, make sure it it there
Thread.Sleep(5 * 1000);
var verifyNewBucket = GraphResponse<Bucket>.Get("buckets/" + newBucket.Id, httpClient).Result;
bucket.Tasks = bucket.Tasks.Reverse().ToArray();
foreach (PlannerTask task in bucket.Tasks)
{
task.PlanId = targetPlanId;
task.BucketId = newBucket.Id;
task.OrderHint = " !";
// assignments contain the users assigned to a task
if (addAssignments)
{
foreach (Assignment assignment in task.Assignments.Values)
{
assignment.OrderHint = " !";
}
}
else
{
task.Assignments = new Dictionary<string, Assignment>();
}
var newTask = GraphResponse<PlannerTask>.Post("tasks", httpClient, task).Result;
// remember new task id for next loop
task.Id = newTask.Id;
}
// if we are too quick the created tasks are not available yet
Thread.Sleep(5 * 1000);
foreach (PlannerTask task in bucket.Tasks)
{
var newTaskDetailsResponse = GraphResponse<TaskDetailResponse>.Get("tasks/" + task.Id + "/details", httpClient).Result;
foreach (var checklist in task.TaskDetail.Checklist.Values)
{
checklist.OrderHint = " !";
}
foreach (var reference in task.TaskDetail.References.Values)
{
// same as order hint
reference.PreviewPriority = " !";
}
var updatedTaskDetailsResponse = GraphResponse<TaskDetailResponse>.Patch("tasks/" + task.Id + "/details", httpClient, task.TaskDetail, newTaskDetailsResponse.OdataEtag).Result;
}
}
public static void ExportToCSV()
{
// export the plan
Console.WriteLine("Select the plan(s) to export");
Plan[] exportedPlans = Export(false, true, false);
// convert the plan to CSV
StringWriter csvString = new StringWriter();
csvString.WriteLine("Plan;Bucket;Task;Zugewiesen an;Fällig am;Erledigt am;Erledigt von;Erstellt am;Erstellt von");
foreach (Plan exportedPlan in exportedPlans)
{
foreach (var bucket in exportedPlan.Buckets)
{
foreach (var task in bucket.Tasks)
{
var completedAt = "";
if (task.CompletedDateTime != null)
{
completedAt = task.CompletedDateTime.ToString();
}
var completedBy = GetUserForEdBy(task.CompletedBy) ?? "--";
var dueAt = "";
if (task.DueDateTime != null)
{
dueAt = task.DueDateTime.ToString();
}
var createdBy = GetUserForEdBy(task.CreatedBy) ?? "--";
var assignedTo = GetAssigned(task.Assignments);
if (task.CompletedDateTime == null) { }
csvString.WriteLine($"{exportedPlan.Title};{bucket.Name};{task.Title};{assignedTo};{dueAt};{completedAt};{completedBy};{task.CreatedDateTime};{createdBy}");
}
}
}
var filename = "ExportedPlans.csv";
if (exportedPlans.Length == 1) filename = exportedPlans[0].Title + ".csv";
File.WriteAllText(filename, csvString.ToString());
}
public static void ForgetCredentials()
{
AuthenticationContext ctx = new AuthenticationContext("https://login.microsoftonline.com/common");
ctx.TokenCache.Clear();
Console.ForegroundColor = ConsoleColor.Green;
Console.WriteLine("Vergessen.");
}
private static Bucket SelectBucket()
{
Plan exportedPlan = Export(false).FirstOrDefault();
for (int i = 0; i < exportedPlan.Buckets.Length; i++)
Console.WriteLine("(" + i + ") " + exportedPlan.Buckets[i].Name);
string selectedBucketS = Program.GetInput("Which bucket do you want to use: ");
int selectedBucket = -1;
if (int.TryParse(selectedBucketS, out selectedBucket))
{
return exportedPlan.Buckets[selectedBucket];
}
throw new Exception("Please select a bucket");
}
// allows the user to search for a group, select the right one and then select the right plan
// idea: if only one group matches or only one plan is in the group, that could be preselected
private static Plan[] SelectPlan(bool allowMultiSelect)
{
using (var httpClient = PrepareGroupsClient())
{
bool foundGroup = false;
while (!foundGroup)
{
string groupSearch = Program.GetInput("Please enter the start of the name of the group containing your plan: ");
var groups = GraphResponse<GroupResponse>.Get("?$filter=groupTypes/any(c:c+eq+'Unified') and startswith(displayName, '" + groupSearch + "')", httpClient).Result.Groups;
if (groups.Length == 0)
{
Console.WriteLine("Found no matching group");
}
else
{
foundGroup = true;
Console.WriteLine("Select group:");
for (int i = 0; i < groups.Length; i++)
{
Console.WriteLine("(" + i + ") " + groups[i].DisplayName);
}
string selectedGroupS = Program.GetInput("Which group do you want to use: ");
int selectedGroup = -1;
if (int.TryParse(selectedGroupS, out selectedGroup))
{
var plans = GraphResponse<PlanResponse>.Get(groups[selectedGroup].Id + "/planner/plans", httpClient).Result.Plans;
Console.WriteLine("Select plan:");
for (int i = 0; i < plans.Length; i++)
{
Console.WriteLine("(" + i + ") " + plans[i].Title);
}
if (allowMultiSelect)
Console.WriteLine("(" + plans.Length + ") All plans");
string selectedPlanS = Program.GetInput("Which plan do you want to use: ");
int selectedPlan = -1;
if (int.TryParse(selectedPlanS, out selectedPlan))
{
if (selectedPlan == plans.Length)
return plans;
else
return new Plan[] { plans[selectedPlan] };
}
}
}
}
}
throw new Exception("Please select a plan");
}
private static HttpClient PreparePlannerClient()
{
return PrepareClient(PLANNER_SUB);
}
private static HttpClient PrepareGroupsClient()
{
return PrepareClient(GROUPS_SUB);
}
private static HttpClient PrepareUsersClient()
{
return PrepareClient(USERS_SUB);
}
private static HttpClient PrepareClient(string sub)
{
var token = GetToken().Result;
var httpClient = new HttpClient();
httpClient.DefaultRequestHeaders.Authorization = new AuthenticationHeaderValue("Bearer", token.AccessToken);
httpClient.BaseAddress = new Uri(GRAPH_ENDPOINT + sub);
return httpClient;
}
// see https://github.com/Azure-Samples/active-directory-dotnet-deviceprofile
private static async Task<AuthenticationResult> GetToken()
{
AuthenticationContext ctx = new AuthenticationContext("https://login.microsoftonline.com/" + TENANT, true);
if (OperatingSystem.IsWindows()) // EncryptedFileCache only works on Windows
{
ctx = new AuthenticationContext("https://login.microsoftonline.com/" + TENANT, true, new EncryptedFileCache());
}
AuthenticationResult result = null;
try
{
result = await ctx.AcquireTokenSilentAsync(RESOURCE_ID, CLIENT_ID);
}
catch (AdalSilentTokenAcquisitionException)
{
result = await GetTokenViaCode(ctx);
}
return result;
}
static async Task<AuthenticationResult> GetTokenViaCode(AuthenticationContext ctx)
{
AuthenticationResult result = null;
DeviceCodeResult codeResult = await ctx.AcquireDeviceCodeAsync(RESOURCE_ID, CLIENT_ID);
Console.ResetColor();
Console.WriteLine("You need to sign in.");
Console.WriteLine("Message: " + codeResult.Message + "\n");
result = await ctx.AcquireTokenByDeviceCodeAsync(codeResult);
return result;
}
private static string GetAssigned(Dictionary<string, Assignment> assignments)
{
if (assignments.Values.Count > 0)
{
var sb = new StringBuilder();
var delim = "";
List<string> added = new List<string>();
foreach (string assignment in assignments.Keys)
{
var user = GetUserForId(assignment) ?? "--";
if (!added.Contains(user))
{
sb.Append(delim);
sb.Append(user);
delim = ", ";
added.Add(user);
}
}
return sb.ToString();
}
else
{
return "--";
}
}
private static string GetUserForEdBy(EdBy edBy)
{
if (edBy == null || edBy.User == null)
return null;
var id = edBy.User.Id;
if (users.ContainsKey(id))
return users[id];
else
{
var user = GetUserForId(id);
return GetUserForId(id);
}
}
private static string GetUserForId(string id)
{
if (id == null)
return null;
if (users.ContainsKey(id))
return users[id];
else
{
using (var httpClient = PrepareUsersClient())
{
try
{
var user = GraphResponse<UserResponse>.Get(id, httpClient).Result;
users.Add(id, user.DisplayName);
return user.DisplayName;
}
catch
{
return null;
}
}
}
}
}
// thanks to https://blog.mariusschulz.com/2017/02/28/detecting-the-operating-system-in-net-core
public static class OperatingSystem
{
public static bool IsWindows() =>
RuntimeInformation.IsOSPlatform(OSPlatform.Windows);
public static bool IsMacOS() =>
RuntimeInformation.IsOSPlatform(OSPlatform.OSX);
public static bool IsLinux() =>
RuntimeInformation.IsOSPlatform(OSPlatform.Linux);
}
}