-
Notifications
You must be signed in to change notification settings - Fork 8.1k
Add Unblock-File for macOS
#11137
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Add Unblock-File for macOS
#11137
Changes from all commits
31be778
a98e089
ef87a3d
088dd3b
ad12b34
7823022
a3c7d80
8adccdc
9f16948
760aa10
d4f9392
da3da7c
d7c5a17
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
| Original file line number | Diff line number | Diff line change |
|---|---|---|
| @@ -1,18 +1,19 @@ | ||
| // Copyright (c) Microsoft Corporation. All rights reserved. | ||
TravisEz13 marked this conversation as resolved.
Show resolved
Hide resolved
TravisEz13 marked this conversation as resolved.
Show resolved
Hide resolved
TravisEz13 marked this conversation as resolved.
Show resolved
Hide resolved
TravisEz13 marked this conversation as resolved.
Show resolved
Hide resolved
TravisEz13 marked this conversation as resolved.
Show resolved
Hide resolved
TravisEz13 marked this conversation as resolved.
Show resolved
Hide resolved
TravisEz13 marked this conversation as resolved.
Show resolved
Hide resolved
|
||
| // Licensed under the MIT License. | ||
|
|
||
| #if !UNIX | ||
|
|
||
| #region Using directives | ||
|
|
||
| using System; | ||
| using System.Collections.Generic; | ||
| using System.Collections.ObjectModel; | ||
| using System.ComponentModel; | ||
| using System.Diagnostics; | ||
| using System.Diagnostics.CodeAnalysis; | ||
| using System.Globalization; | ||
| using System.IO; | ||
| using System.Management.Automation; | ||
| using System.Management.Automation.Internal; | ||
| using System.Runtime.InteropServices; | ||
|
|
||
| #endregion | ||
|
|
||
|
|
@@ -23,6 +24,11 @@ namespace Microsoft.PowerShell.Commands | |
| HelpUri = "https://go.microsoft.com/fwlink/?LinkID=217450")] | ||
| public sealed class UnblockFileCommand : PSCmdlet | ||
| { | ||
| #if UNIX | ||
| private const string MacBlockAttribute = "com.apple.quarantine"; | ||
| private const int RemovexattrFollowSymLink = 0; | ||
| #endif | ||
|
|
||
| /// <summary> | ||
| /// The path of the file to unblock. | ||
| /// </summary> | ||
|
|
@@ -112,6 +118,7 @@ protected override void ProcessRecord() | |
| } | ||
| } | ||
| } | ||
| #if !UNIX | ||
|
|
||
| // Unblock files | ||
| foreach (string path in pathsToProcess) | ||
|
|
@@ -124,10 +131,34 @@ protected override void ProcessRecord() | |
| } | ||
| catch (Exception e) | ||
| { | ||
| WriteError(new ErrorRecord(e, "RemoveItemUnableToAccessFile", ErrorCategory.ResourceUnavailable, path)); | ||
| WriteError(new ErrorRecord(exception: e, errorId: "RemoveItemUnableToAccessFile", ErrorCategory.ResourceUnavailable, targetObject: path)); | ||
| } | ||
| } | ||
| } | ||
| #else | ||
| if (RuntimeInformation.IsOSPlatform(OSPlatform.Linux)) | ||
| { | ||
| string errorMessage = UnblockFileStrings.LinuxNotSupported; | ||
| Exception e = new PlatformNotSupportedException(errorMessage); | ||
| ThrowTerminatingError(new ErrorRecord(exception: e, errorId: "LinuxNotSupported", ErrorCategory.NotImplemented, targetObject: null)); | ||
| return; | ||
| } | ||
|
|
||
| foreach (string path in pathsToProcess) | ||
| { | ||
| if(IsBlocked(path)) | ||
TravisEz13 marked this conversation as resolved.
Show resolved
Hide resolved
|
||
| { | ||
| UInt32 result = RemoveXattr(path, MacBlockAttribute, RemovexattrFollowSymLink); | ||
| if(result != 0) | ||
| { | ||
| string errorMessage = string.Format(CultureInfo.CurrentUICulture, UnblockFileStrings.UnblockError, path); | ||
| Exception e = new InvalidOperationException(errorMessage); | ||
|
Member
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. maybe this should be
Member
Author
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. definitely not, we are on a supported platform here.
Member
Author
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. maybe you meant line 142... changed it there. |
||
| WriteError(new ErrorRecord(exception: e, errorId: "UnblockError", ErrorCategory.InvalidResult, targetObject: path)); | ||
| } | ||
| } | ||
| } | ||
|
|
||
| #endif | ||
| } | ||
|
|
||
| /// <summary> | ||
|
|
@@ -163,6 +194,36 @@ private bool IsValidFileForUnblocking(string resolvedpath) | |
|
|
||
| return isValidUnblockableFile; | ||
| } | ||
|
|
||
| #if UNIX | ||
| private bool IsBlocked(string path) | ||
| { | ||
| uint valueSize = 1024; | ||
| IntPtr value = Marshal.AllocHGlobal((int)valueSize); | ||
| try | ||
| { | ||
| var resultSize = GetXattr(path, MacBlockAttribute, value, valueSize, 0, RemovexattrFollowSymLink); | ||
| return resultSize != -1; | ||
| } | ||
| finally | ||
| { | ||
| Marshal.FreeHGlobal(value); | ||
| } | ||
| } | ||
|
|
||
| // Ansi means UTF8 on Unix | ||
| // https://developer.apple.com/library/archive/documentation/System/Conceptual/ManPages_iPhoneOS/man2/RemoveXattr.2.html | ||
| [DllImport("libc", SetLastError = true, EntryPoint = "removexattr", CharSet = CharSet.Ansi)] | ||
| private static extern UInt32 RemoveXattr(string path, string name, int options); | ||
|
|
||
| [DllImport("libc", EntryPoint = "getxattr", CharSet = CharSet.Ansi)] | ||
| private static extern long GetXattr( | ||
| [MarshalAs(UnmanagedType.LPStr)] string path, | ||
| [MarshalAs(UnmanagedType.LPStr)] string name, | ||
| IntPtr value, | ||
| ulong size, | ||
| uint position, | ||
| int options); | ||
| #endif | ||
| } | ||
| } | ||
| #endif | ||
| Original file line number | Diff line number | Diff line change |
|---|---|---|
| @@ -0,0 +1,126 @@ | ||
| <?xml version="1.0" encoding="utf-8"?> | ||
| <root> | ||
| <!-- | ||
| Microsoft ResX Schema | ||
|
|
||
| Version 2.0 | ||
|
|
||
| The primary goals of this format is to allow a simple XML format | ||
| that is mostly human readable. The generation and parsing of the | ||
| various data types are done through the TypeConverter classes | ||
| associated with the data types. | ||
|
|
||
| Example: | ||
|
|
||
| ... ado.net/XML headers & schema ... | ||
| <resheader name="resmimetype">text/microsoft-resx</resheader> | ||
| <resheader name="version">2.0</resheader> | ||
| <resheader name="reader">System.Resources.ResXResourceReader, System.Windows.Forms, ...</resheader> | ||
| <resheader name="writer">System.Resources.ResXResourceWriter, System.Windows.Forms, ...</resheader> | ||
| <data name="Name1"><value>this is my long string</value><comment>this is a comment</comment></data> | ||
| <data name="Color1" type="System.Drawing.Color, System.Drawing">Blue</data> | ||
| <data name="Bitmap1" mimetype="application/x-microsoft.net.object.binary.base64"> | ||
| <value>[base64 mime encoded serialized .NET Framework object]</value> | ||
| </data> | ||
| <data name="Icon1" type="System.Drawing.Icon, System.Drawing" mimetype="application/x-microsoft.net.object.bytearray.base64"> | ||
| <value>[base64 mime encoded string representing a byte array form of the .NET Framework object]</value> | ||
| <comment>This is a comment</comment> | ||
| </data> | ||
|
|
||
| There are any number of "resheader" rows that contain simple | ||
| name/value pairs. | ||
|
|
||
| Each data row contains a name, and value. The row also contains a | ||
| type or mimetype. Type corresponds to a .NET class that support | ||
| text/value conversion through the TypeConverter architecture. | ||
| Classes that don't support this are serialized and stored with the | ||
| mimetype set. | ||
|
|
||
| The mimetype is used for serialized objects, and tells the | ||
| ResXResourceReader how to depersist the object. This is currently not | ||
| extensible. For a given mimetype the value must be set accordingly: | ||
|
|
||
| Note - application/x-microsoft.net.object.binary.base64 is the format | ||
| that the ResXResourceWriter will generate, however the reader can | ||
| read any of the formats listed below. | ||
|
|
||
| mimetype: application/x-microsoft.net.object.binary.base64 | ||
| value : The object must be serialized with | ||
| : System.Runtime.Serialization.Formatters.Binary.BinaryFormatter | ||
| : and then encoded with base64 encoding. | ||
|
|
||
| mimetype: application/x-microsoft.net.object.soap.base64 | ||
| value : The object must be serialized with | ||
| : System.Runtime.Serialization.Formatters.Soap.SoapFormatter | ||
| : and then encoded with base64 encoding. | ||
|
|
||
| mimetype: application/x-microsoft.net.object.bytearray.base64 | ||
| value : The object must be serialized into a byte array | ||
| : using a System.ComponentModel.TypeConverter | ||
| : and then encoded with base64 encoding. | ||
| --> | ||
| <xsd:schema id="root" xmlns="" xmlns:xsd="https://www.w3.org/2001/XMLSchema" xmlns:msdata="urn:schemas-microsoft-com:xml-msdata"> | ||
| <xsd:import namespace="https://www.w3.org/XML/1998/namespace" /> | ||
| <xsd:element name="root" msdata:IsDataSet="true"> | ||
| <xsd:complexType> | ||
| <xsd:choice maxOccurs="unbounded"> | ||
| <xsd:element name="metadata"> | ||
| <xsd:complexType> | ||
| <xsd:sequence> | ||
| <xsd:element name="value" type="xsd:string" minOccurs="0" /> | ||
| </xsd:sequence> | ||
| <xsd:attribute name="name" use="required" type="xsd:string" /> | ||
| <xsd:attribute name="type" type="xsd:string" /> | ||
| <xsd:attribute name="mimetype" type="xsd:string" /> | ||
| <xsd:attribute ref="xml:space" /> | ||
| </xsd:complexType> | ||
| </xsd:element> | ||
| <xsd:element name="assembly"> | ||
| <xsd:complexType> | ||
| <xsd:attribute name="alias" type="xsd:string" /> | ||
| <xsd:attribute name="name" type="xsd:string" /> | ||
| </xsd:complexType> | ||
| </xsd:element> | ||
| <xsd:element name="data"> | ||
| <xsd:complexType> | ||
| <xsd:sequence> | ||
| <xsd:element name="value" type="xsd:string" minOccurs="0" msdata:Ordinal="1" /> | ||
| <xsd:element name="comment" type="xsd:string" minOccurs="0" msdata:Ordinal="2" /> | ||
| </xsd:sequence> | ||
| <xsd:attribute name="name" type="xsd:string" use="required" msdata:Ordinal="1" /> | ||
| <xsd:attribute name="type" type="xsd:string" msdata:Ordinal="3" /> | ||
| <xsd:attribute name="mimetype" type="xsd:string" msdata:Ordinal="4" /> | ||
| <xsd:attribute ref="xml:space" /> | ||
| </xsd:complexType> | ||
| </xsd:element> | ||
| <xsd:element name="resheader"> | ||
| <xsd:complexType> | ||
| <xsd:sequence> | ||
| <xsd:element name="value" type="xsd:string" minOccurs="0" msdata:Ordinal="1" /> | ||
| </xsd:sequence> | ||
| <xsd:attribute name="name" type="xsd:string" use="required" /> | ||
| </xsd:complexType> | ||
| </xsd:element> | ||
| </xsd:choice> | ||
| </xsd:complexType> | ||
| </xsd:element> | ||
| </xsd:schema> | ||
| <resheader name="resmimetype"> | ||
| <value>text/microsoft-resx</value> | ||
| </resheader> | ||
| <resheader name="version"> | ||
| <value>2.0</value> | ||
| </resheader> | ||
| <resheader name="reader"> | ||
| <value>System.Resources.ResXResourceReader, System.Windows.Forms, Version=2.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value> | ||
| </resheader> | ||
| <resheader name="writer"> | ||
| <value>System.Resources.ResXResourceWriter, System.Windows.Forms, Version=2.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value> | ||
| </resheader> | ||
| <data name="LinuxNotSupported" xml:space="preserve"> | ||
| <value>The cmdlet does not support Linux.</value> | ||
| </data> | ||
| <data name="UnblockError" xml:space="preserve"> | ||
| <value>There was an error unblocking {0}.</value> | ||
| </data> | ||
| </root> |
Uh oh!
There was an error while loading. Please reload this page.