// Copyright (C) Stichting Deltares 2016. All rights reserved. // // This file is part of Ringtoets. // // Ringtoets is free software: you can redistribute it and/or modify // it under the terms of the GNU Lesser General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU Lesser General Public License for more details. // // You should have received a copy of the GNU Lesser General Public License // along with this program. If not, see . // // All names, logos, and references to "Deltares" are registered trademarks of // Stichting Deltares and remain full property of Stichting Deltares at all times. // All rights reserved. using System; using System.Collections.Generic; using System.IO; using System.Linq; using System.Security.AccessControl; using System.Security.Principal; namespace Core.Common.TestUtil { /// /// Class for providing a safe way to manipulating directory permissions. /// public class DirectoryPermissionsRevoker : IDisposable { private readonly IList appliedFileSystemAccessRules = new List(); private readonly string folderPath; private readonly DirectoryInfo directoryInfo; /// /// Creates an instance of . /// Adds a of type to the access /// rule set for the folder at . /// /// The path of the file to change the right for. /// The right to deny. public DirectoryPermissionsRevoker(string folderPath, FileSystemRights rights) { if (string.IsNullOrWhiteSpace(folderPath)) { throw new ArgumentException(@"folderPath must have a valid value.", "folderPath"); } this.folderPath = folderPath; directoryInfo = new DirectoryInfo(Path.GetFullPath(folderPath)); if (!directoryInfo.Exists) { throw new DirectoryNotFoundException(@"folderPath does not exist."); } if ((rights ^ FileSystemRights.Synchronize) == 0) { // The FileSystemRights Synchronize by itself cannot be set. throw new NotSupportedException(string.Format("Setting the right {0} is not supported.", rights)); } AddDenyDirectoryInfoRight(GetSupportedFileSystemRights(rights)); } public void Dispose() { if (!Directory.Exists(folderPath)) { return; } foreach (var appliedFileSystemAccessRule in appliedFileSystemAccessRules) { RevertDenyDirectoryInfoRight(appliedFileSystemAccessRule); } } private void AddDenyDirectoryInfoRight(FileSystemRights rights) { DirectorySecurity directorySecurity = directoryInfo.GetAccessControl(); if (IsFileSystemAccessRuleSet(rights, directorySecurity, AccessControlType.Deny)) { return; } SecurityIdentifier sid = GetSecurityIdentifier(); var fileSystemAccessRule = new FileSystemAccessRule(sid, rights, AccessControlType.Deny); directorySecurity.AddAccessRule(fileSystemAccessRule); directoryInfo.SetAccessControl(directorySecurity); appliedFileSystemAccessRules.Add(fileSystemAccessRule); } private static bool IsFileSystemAccessRuleSet(FileSystemRights rights, CommonObjectSecurity commonObjectSecurity, AccessControlType accessControlType) { AuthorizationRuleCollection rules = commonObjectSecurity.GetAccessRules(true, false, typeof(SecurityIdentifier)); return rules.OfType().Any(fs => fs.FileSystemRights.HasFlag(rights) && fs.AccessControlType == accessControlType); } private static FileSystemRights GetSupportedFileSystemRights(FileSystemRights rights) { return rights & ~FileSystemRights.Synchronize; } private void RevertDenyDirectoryInfoRight(FileSystemAccessRule rule) { DirectorySecurity directorySecurity = directoryInfo.GetAccessControl(); directorySecurity.RemoveAccessRule(rule); directoryInfo.SetAccessControl(directorySecurity); } private static SecurityIdentifier GetSecurityIdentifier() { SecurityIdentifier id = WindowsIdentity.GetCurrent().User.AccountDomainSid; return new SecurityIdentifier(WellKnownSidType.WorldSid, id); } } }