-
Notifications
You must be signed in to change notification settings - Fork 200
/
Copy pathSnapshot.cs
409 lines (378 loc) · 19.9 KB
/
Snapshot.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
#if !(NET20 || NET35 || NET40)
using System;
using System.Collections.Generic;
using System.Globalization;
using System.IO;
using System.IO.Compression;
using System.Linq;
using System.Security.AccessControl;
using System.Text;
using System.Xml;
using System.Xml.Schema;
using System.Xml.Serialization;
namespace Microsoft.Win32.TaskScheduler
{
/// <summary>Abstract class representing a secured item for storage in a <see cref="TaskSchedulerSnapshot"/>.</summary>
public abstract class SnapshotItem
{
/// <summary>Initializes a new instance of the <see cref="SnapshotItem"/> class.</summary>
/// <param name="path">The path to the item.</param>
/// <param name="sddl">The SDDL for the item.</param>
internal SnapshotItem(string path, string sddl) { Path = path; Sddl = sddl; }
/// <summary>Gets the path to the item.</summary>
/// <value>The path to the item.</value>
public string Path { get; private set; }
/// <summary>Gets the SDDL for the item.</summary>
/// <value>The SDDL for the item.</value>
public string Sddl { get; private set; }
}
/// <summary>Represents a <see cref="TaskFolder"/> instance and captures its name and security.</summary>
public sealed class TaskFolderSnapshot : SnapshotItem
{
/// <summary>Initializes a new instance of the <see cref="TaskFolderSnapshot"/> class.</summary>
/// <param name="path">The path to the item.</param>
/// <param name="sddl">The SDDL for the item.</param>
internal TaskFolderSnapshot(string path, string sddl) : base(path, sddl) { }
}
/// <summary>
/// Represents all the information about the tasks and folders from a <see cref="TaskService"/> instance that can be used to reconstitute tasks and folders
/// on the same or different systems. <note>This class and related classes are only available under the .NET 4.5.2 build and later .NET versions due to
/// dependencies on threading and compressed (zip) files.</note>
/// </summary>
public sealed class TaskSchedulerSnapshot : IXmlSerializable
{
private const string hdrfile = ".metadata.xml";
private List<SnapshotItem> items;
/// <summary>Creates a new instance of <see cref="TaskSchedulerSnapshot"/> from an existing snapshot.</summary>
/// <param name="path">The zip file snapshot created by the <see cref="Create(TaskService,string)"/> method.</param>
public TaskSchedulerSnapshot(string path)
{
if (path == null) throw new ArgumentNullException(nameof(path));
if (!File.Exists(path)) throw new FileNotFoundException("Invalid file location.", nameof(path));
try
{
using (var zip = ZipFile.OpenRead(path))
zip.GetEntry(hdrfile);
}
catch
{
throw new InvalidOperationException("Invalid file format.");
}
Path = path;
}
internal TaskSchedulerSnapshot() { }
/// <summary>
/// Gets a list of <see cref="TaskSnapshot"/> and <see cref="TaskFolderSnapshot"/> instances the represent the tasks and folders from a Task Scheduler instance.
/// </summary>
public List<SnapshotItem> Items
{
get => items ?? (items = (Path == null ? new List<SnapshotItem>() : GetArchiveItems(Path)));
internal set => items = value;
}
/// <summary>Gets the path of the file based snapshot.</summary>
public string Path { get; private set; }
/// <summary>Gets the machine name of the server from which the snapshot was taken.</summary>
/// <value>The target server name.</value>
public string TargetServer { get; private set; }
/// <summary>Gets the UTC time stamp for when the snapshot was taken.</summary>
/// <value>The time stamp.</value>
public DateTime TimeStamp { get; private set; } = DateTime.UtcNow;
/// <summary>
/// Creates a compressed zip file that contains all the information accessible to the user from the <see cref="TaskService"/> instance necessary to
/// reconstitute its tasks and folders. <note>This method can take many seconds to execute. It is recommended to call the asynchronous
/// version.</note><note type="warning">This method will execute without error even if the user does not have permissions to see all tasks and folders.
/// It is imperative that the developer ensures that the user has Administrator or equivalent rights before calling this method.</note>
/// </summary>
/// <param name="ts">The <see cref="TaskService"/> from which to pull the tasks and folders.</param>
/// <param name="path">The output zip file in which to place the snapshot information.</param>
/// <returns>A <see cref="TaskSchedulerSnapshot"/> instance with the contents of the specified Task Scheduler connection.</returns>
public static TaskSchedulerSnapshot Create(TaskService ts, string path)
{
var c = new System.Threading.CancellationTokenSource();
return InternalCreate(ts.Token, path, c.Token, null);
}
/// <summary>
/// Creates a compressed zip file that contains all the information accessible to the user from the <see cref="TaskService"/> instance necessary to
/// reconstitute its tasks and folders. <note type="warning">This method will execute without error even if the user does not have permissions to see all
/// tasks and folders. It is imperative that the developer ensures that the user has Administrator or equivalent rights before calling this method.</note>
/// </summary>
/// <param name="tsToken">The <see cref="TaskService.ConnectionToken"/> from which to pull the tasks and folders.</param>
/// <param name="path">The output zip file in which to place the snapshot information.</param>
/// <param name="cancelToken">A cancellation token to use to cancel this asynchronous operation.</param>
/// <param name="progress">An optional <see cref="IProgress{T}"/> instance to use to report progress of the asynchronous operation.</param>
/// <returns>An asynchronous <see cref="TaskSchedulerSnapshot"/> instance with the contents of the specified Task Scheduler connection.</returns>
public static async System.Threading.Tasks.Task<TaskSchedulerSnapshot> Create(TaskService.ConnectionToken tsToken, string path, System.Threading.CancellationToken cancelToken, IProgress<Tuple<int, string>> progress)
{
return await System.Threading.Tasks.Task.Run(() => InternalCreate(tsToken, path, cancelToken, progress), cancelToken);
}
/// <summary>Opens an existing snapshot and returns a new instance of <see cref="TaskSchedulerSnapshot"/>.</summary>
/// <param name="path">The zip file snapshot created by the <see cref="Create(TaskService,string)"/> method.</param>
/// <returns>A <see cref="TaskSchedulerSnapshot"/> instance with the contents of the specified snapshot file.</returns>
public static TaskSchedulerSnapshot Open(string path) => new TaskSchedulerSnapshot(path);
/// <summary>Register a list of snapshot items (tasks and folders) into the specified Task Scheduler.</summary>
/// <param name="tsToken">The <see cref="TaskService.ConnectionToken"/> into which the tasks and folders are registered.</param>
/// <param name="itemPaths">
/// The list of paths representing the tasks and folders from this snapshot that should be registered on the <see cref="TaskService"/> instance.
/// </param>
/// <param name="applyAccessRights">
/// If <c>true</c>, takes the access rights from the snapshot item and applies it to both new and existing tasks and folders.
/// </param>
/// <param name="overwriteExisting">
/// If <c>true</c>, overwrite any existing tasks and folders found in the target Task Scheduler that match the path of the snapshot item.
/// </param>
/// <param name="passwords">
/// Lookup table for password. Provide pairs of the user/group account name and the associated passwords for any task that requires a password.
/// </param>
/// <param name="cancelToken">A cancellation token to use to cancel this asynchronous operation.</param>
/// <param name="progress">An optional <see cref="IProgress{T}"/> instance to use to report progress of the asynchronous operation.</param>
/// <returns>An asynchronous <see cref="Task"/> instance.</returns>
public async System.Threading.Tasks.Task Restore(TaskService.ConnectionToken tsToken, IEnumerable<string> itemPaths, bool applyAccessRights, bool overwriteExisting, IDictionary<string, string> passwords, System.Threading.CancellationToken cancelToken, IProgress<Tuple<int, string>> progress)
{
if (itemPaths == null) throw new ArgumentNullException(nameof(itemPaths));
var items = Items.Where(i => i is TaskSnapshot).Join(itemPaths, a => a.Path, b => b, (a, b) => a).ToList();
if (items.Count != itemPaths.Count()) throw new ArgumentException($"Unable to locate matching tasks to all values of {nameof(itemPaths)}.", nameof(itemPaths));
await System.Threading.Tasks.Task.Run(() => InternalRestore(tsToken, items, applyAccessRights, overwriteExisting, passwords, cancelToken, progress));
}
/// <summary>Register a list of snapshot items (tasks and folders) into the specified Task Scheduler.</summary>
/// <param name="tsToken">The <see cref="TaskService.ConnectionToken"/> into which the tasks and folders are registered.</param>
/// <param name="items">
/// The list of <see cref="SnapshotItem"/> instances representing the tasks and folders from this snapshot that should be registered on the
/// <see cref="TaskService"/> instance.
/// </param>
/// <param name="applyAccessRights">
/// If <c>true</c>, takes the access rights from the snapshot item and applies it to both new and existing tasks and folders.
/// </param>
/// <param name="overwriteExisting">
/// If <c>true</c>, overwrite any existing tasks and folders found in the target Task Scheduler that match the path of the snapshot item.
/// </param>
/// <param name="passwords">
/// Lookup table for password. Provide pairs of the user/group account name and the associated passwords for any task that requires a password.
/// </param>
/// <param name="cancelToken">A cancellation token to use to cancel this asynchronous operation.</param>
/// <param name="progress">An optional <see cref="IProgress{T}"/> instance to use to report progress of the asynchronous operation.</param>
/// <returns>An asynchronous <see cref="Task"/> instance.</returns>
public async System.Threading.Tasks.Task Restore(TaskService.ConnectionToken tsToken, ICollection<SnapshotItem> items, bool applyAccessRights, bool overwriteExisting, IDictionary<string, string> passwords, System.Threading.CancellationToken cancelToken, IProgress<Tuple<int, string>> progress)
{
if (items == null) throw new ArgumentNullException(nameof(items));
await System.Threading.Tasks.Task.Run(() => InternalRestore(tsToken, items, applyAccessRights, overwriteExisting, passwords, cancelToken, progress));
}
/// <summary>Register a list of snapshot items (tasks and folders) into the specified Task Scheduler.</summary>
/// <param name="ts">The <see cref="TaskService"/> into which the tasks and folders are registered.</param>
/// <param name="items">
/// The list of <see cref="SnapshotItem"/> instances representing the tasks and folders from this snapshot that should be registered on the
/// <see cref="TaskService"/> instance.
/// </param>
/// <param name="applyAccessRights">
/// If <c>true</c>, takes the access rights from the snapshot item and applies it to both new and existing tasks and folders.
/// </param>
/// <param name="overwriteExisting">
/// If <c>true</c>, overwrite any existing tasks and folders found in the target Task Scheduler that match the path of the snapshot item.
/// </param>
/// <param name="passwords">
/// Lookup table for password. Provide pairs of the user/group account name and the associated passwords for any task that requires a password.
/// </param>
public void Restore(TaskService ts, ICollection<SnapshotItem> items, bool applyAccessRights, bool overwriteExisting, IDictionary<string, string> passwords)
{
var c = new System.Threading.CancellationTokenSource();
InternalRestore(ts.Token, items, applyAccessRights, overwriteExisting, passwords, c.Token, null);
}
XmlSchema IXmlSerializable.GetSchema() => null;
void IXmlSerializable.ReadXml(XmlReader reader)
{
reader.MoveToContent();
if (DateTime.TryParseExact(reader.GetAttribute("timestamp"), "o", CultureInfo.CurrentCulture.DateTimeFormat, DateTimeStyles.AssumeUniversal, out DateTime dt))
TimeStamp = dt;
TargetServer = reader.GetAttribute("machine");
reader.GetAttribute("version");
while (reader.Read())
{
if (reader.Name == "Task")
{
var tsnap = new TaskSnapshot(reader.GetAttribute("path"), reader.GetAttribute("sddl"), true);
var e = reader.GetAttribute("enabled");
if (e != null && e == "false") tsnap.Enabled = false;
Items.Add(tsnap);
}
else if (reader.Name == "TaskFolder")
Items.Add(new TaskFolderSnapshot(reader.GetAttribute("path"), reader.GetAttribute("sddl")));
}
}
void IXmlSerializable.WriteXml(XmlWriter writer)
{
writer.WriteAttributeString("timestamp", TimeStamp.ToString("o"));
if (TargetServer != null)
writer.WriteAttributeString("machine", TargetServer);
writer.WriteAttributeString("version", "1.0");
foreach (var i in items)
{
if (i is TaskSnapshot t)
{
writer.WriteStartElement("Task");
writer.WriteAttributeString("path", t.Path);
writer.WriteAttributeString("sddl", t.Sddl);
if (!t.Enabled)
writer.WriteAttributeString("enabled", "false");
writer.WriteEndElement();
}
else if (i is TaskFolderSnapshot f)
{
writer.WriteStartElement("TaskFolder");
writer.WriteAttributeString("path", f.Path);
writer.WriteAttributeString("sddl", f.Sddl);
writer.WriteEndElement();
}
}
}
private static List<SnapshotItem> GetArchiveItems(string archiveFile)
{
TaskSchedulerSnapshot ret = null;
using (var zip = ZipFile.OpenRead(archiveFile))
{
using (var hdrStr = zip.GetEntry(hdrfile).Open())
ret = new XmlSerializer(typeof(TaskSchedulerSnapshot)).Deserialize(hdrStr) as TaskSchedulerSnapshot;
if (ret == null) return null;
for (int i = 0; i < ret.Items.Count; i++)
{
if (ret.Items[i] is TaskSnapshot t)
{
var xml = zip.GetEntry(t.Path.TrimStart('\\') + ".xml");
using (var str = new StreamReader(xml.Open(), Encoding.UTF8))
t.TaskDefinitionXml = str.ReadToEnd();
}
}
}
return ret?.Items;
}
private static TaskSchedulerSnapshot InternalCreate(TaskService.ConnectionToken token, string path, System.Threading.CancellationToken cancelToken, IProgress<Tuple<int, string>> progress)
{
var ts = TaskService.CreateFromToken(token);
const SecurityInfos siall = SecurityInfos.DiscretionaryAcl | SecurityInfos.SystemAcl | SecurityInfos.Group | SecurityInfos.Owner;
if (File.Exists(path)) throw new ArgumentException("Output file already exists.", nameof(path));
int i = 0, count = 0;
using (var zipstr = new FileStream(path, FileMode.CreateNew))
using (var zip = new ZipArchive(zipstr, ZipArchiveMode.Create))
{
GetCount(ts.RootFolder);
var snapshot = new TaskSchedulerSnapshot { TargetServer = ts.TargetServer ?? Environment.MachineName };
GetContents(ts.RootFolder, snapshot.Items, zip);
snapshot.Items.Sort((t1, t2) => String.Compare(t1.Path, t2.Path, StringComparison.InvariantCultureIgnoreCase));
using (var hdr = new StreamWriter(zip.CreateEntry(hdrfile).Open(), Encoding.UTF8))
new XmlSerializer(snapshot.GetType()).Serialize(hdr, snapshot);
snapshot.Path = path;
return snapshot;
}
void GetCount(TaskFolder f)
{
count += f.Tasks.Count;
foreach (var sf in f.SubFolders)
{
count++;
GetCount(sf);
}
}
void GetContents(TaskFolder f, List<SnapshotItem> list, ZipArchive zip)
{
cancelToken.ThrowIfCancellationRequested();
foreach (var t in f.Tasks)
{
list.Add(new TaskSnapshot(t.Path, t.GetSecurityDescriptorSddlForm(siall), t.Enabled, t.Xml));
using (var wr = new StreamWriter(zip.CreateEntry(t.Path.TrimStart('\\') + ".xml").Open(), Encoding.Unicode))
wr.Write(t.Xml);
cancelToken.ThrowIfCancellationRequested();
progress?.Report(new Tuple<int, string>(++i * 100 / count, t.Path));
}
foreach (var sf in f.SubFolders)
{
list.Add(new TaskFolderSnapshot(sf.Path, sf.GetSecurityDescriptorSddlForm(siall)));
zip.CreateEntry(sf.Path.TrimStart('\\') + "\\");
GetContents(sf, list, zip);
cancelToken.ThrowIfCancellationRequested();
progress?.Report(new Tuple<int, string>(++i * 100 / count, sf.Path));
}
}
}
private void InternalRestore(TaskService.ConnectionToken token, ICollection<SnapshotItem> items, bool applyAccessRights, bool overwriteExisting, IDictionary<string, string> passwords, System.Threading.CancellationToken cancelToken, IProgress<Tuple<int, string>> progress)
{
var ts = TaskService.CreateFromToken(token);
var i = 0;
progress?.Report(new Tuple<int, string>(0, ""));
foreach (var item in items)
{
cancelToken.ThrowIfCancellationRequested();
if (item is TaskSnapshot t)
{
var td = ts.NewTask();
td.XmlText = t.TaskDefinitionXml;
var pwd = td.Principal.RequiresPassword() ? passwords?[td.Principal.ToString()] : null;
var st = ts.GetTask(t.Path);
if (st == null)
{
CreateTask(t, td, pwd);
}
else if (overwriteExisting)
{
st = st.Folder.RegisterTaskDefinition(System.IO.Path.GetFileName(t.Path), td, TaskCreation.CreateOrUpdate, td.Principal.ToString(), pwd, td.Principal.LogonType, applyAccessRights ? t.Sddl : null);
if (!t.Enabled) st.Enabled = false;
}
}
else if (item is TaskFolderSnapshot f)
{
var sf = ts.GetFolder(f.Path);
if (sf == null)
sf = EnsureFolder(f.Path);
else if (overwriteExisting && applyAccessRights)
sf.SetSecurityDescriptorSddlForm(f.Sddl);
}
progress?.Report(new Tuple<int, string>(++i * 100 / items.Count, item.Path));
}
progress?.Report(new Tuple<int, string>(100, ""));
void CreateTask(TaskSnapshot task, TaskDefinition td, string password)
{
var fpath = System.IO.Path.GetDirectoryName(task.Path);
var fld = EnsureFolder(fpath);
var t = fld.RegisterTaskDefinition(System.IO.Path.GetFileName(task.Path), td, TaskCreation.CreateOrUpdate, td.Principal.ToString(), password, td.Principal.LogonType, applyAccessRights ? task.Sddl : null);
if (!task.Enabled) t.Enabled = false;
}
TaskFolder EnsureFolder(string fpath)
{
if (!overwriteExisting)
{
var f = ts.GetFolder(fpath);
if (f != null) return f;
}
var flds = fpath.Split(new[] { '\\' }, StringSplitOptions.RemoveEmptyEntries);
var fld = ts.RootFolder;
var sfpath = "";
for (var j = 0; j < flds.Length; j++)
{
sfpath += "\\" + flds[j];
var sf = fld.SubFolders.Exists(flds[j]) ? fld.SubFolders[flds[j]] : null;
if (sf == null)
sf = fld.CreateFolder(flds[j], GetFolderSddl(sfpath), false);
else if (overwriteExisting)
sf.SetSecurityDescriptorSddlForm(GetFolderSddl(sfpath));
fld = sf;
}
return fld;
}
string GetFolderSddl(string path) => Items?.Find(tci => String.Equals(tci.Path, path, StringComparison.OrdinalIgnoreCase)).Sddl;
}
}
/// <summary>Represents a <see cref="Task"/> instance and captures its details.</summary>
public sealed class TaskSnapshot : SnapshotItem
{
/// <summary>Initializes a new instance of the <see cref="TaskSnapshot"/> class.</summary>
/// <param name="path">The path to the item.</param>
/// <param name="sddl">The SDDL for the item.</param>
/// <param name="enabled">If set to <c>true</c> task is enabled.</param>
/// <param name="xml">The XML for the <see cref="TaskDefinition"/>.</param>
internal TaskSnapshot(string path, string sddl, bool enabled, string xml = null) : base(path, sddl) { Enabled = enabled; TaskDefinitionXml = xml; }
/// <summary>Gets a value indicating whether th <see cref="Task"/> is enabled.</summary>
/// <value><c>true</c> if enabled; otherwise, <c>false</c>.</value>
public bool Enabled { get; internal set; } = true;
/// <summary>Gets the <see cref="TaskDefinition"/> XML.</summary>
/// <value>The <see cref="TaskDefinition"/> XML.</value>
public string TaskDefinitionXml { get; internal set; }
}
}
#endif