We're headed to GridSecCon 2024, October 22-25 in Minneapolis, MN! Learn more here
Schedule a Demo
Blog August 20, 2020 PowerShell, PSPKI

PowerShell PKI (PSPKI) 3.7 enhancements – Certification Authority API (part 1)

by Vadims Podāns

Hello S-1-1-0, here is another blog post in the PSPKI v3.7 enhancement blog series. Today I will talk about another interesting work done in PSPKI.

Intro

In one of my previous posts, I’ve outlined a very general roadmap for PKI library and the move towards the .NET Core. There are several obstacles for the move and biggest one is tight coupling with COM interfaces.

  • About 146% of Microsoft ADCS API is implemented as COM interfaces which are not good for any .NET Core project. It is very Windows-specific technology and can match Win32 interop. Neither Win32, nor COM are easy to use directly in .NET/PowerShell projects. In this version, I’ve started a work on abstracting (decoupling) internal code from direct access to COM stuff by adding another layer of abstraction using .NET interfaces.
  • Documentation. MSDN documentation on ADCS API is very generic and unclear in most cases. There is better documentation for Distributed COM (DCOM) interfaces published by Open Specifications team.

ADCS API are implemented as low-level DCOM (where server and client can run on different computers) which are very hard to use are not available from managed platform such as .NET. For that purposes are used simplified COM wrappers around DCOM. In some ways, DCOM and COM interface members match (though, not necessary), so the documentation can be reused to some extent. By naming convention, DCOM interfaces are ended with D letter. For example, main ADCS management interface is ICertAdminD and corresponding client-side COM wrapper is ICertAdmin.

  • COM object handling and memory leaks. Once you instantiate a COM object, its reference count is increased (via IUnknown base interface) behind CLR runtime and not visible to CLR garbage collector. This may lead to memory leaks and you have to explicitly dispose COM object from memory when it is no longer needed.

However, during research and countless of tests I found that even this documentation is misleading or otherwise incorrect and I worked with docs team to fix docs issues I found:However, during research and countless of tests I found that even this documentation is misleading or otherwise incorrect and I worked with docs team to fix docs issues I found:

What’s new?

PSPKI actively uses COM interfaces as helpers to perform various ADCS management operations. In v3.7, I refactored the code and moved helpers to individual .NET interfaces and classes. There is a brand new namespace called SysadminsLV.PKI.Dcom. This namespace contains adapted managed interfaces and enumerations for COM interfaces. Every interface that is supposed to be a wrapper around DCOM gets D letter as well. This denotes the purpose of interface and avoids possible interface name duplicates (such as ICertConfig COM interface). COM-based implementations for these interfaces are located in SysadminsLV.PKI.Dcom.Implementations namespace. There can be non COM-based implementations. For example, ICertRegManagerD is implemented in COM-based class: CertSrvRegManagerD and in remote registry-based class: CertSrvRegManager.

For all managed implementation COM objects are disposed automatically when necessary.

Certification Authority autodiscovery

The most useful interface we start with is ICertConfigD interface which is an improved wrapper around ICertConfig COM interface. Implementations of this interface are responsible for Certification Authority discovery and basic information retrieval from registration sources. This interface doesn’t check for connectivity and only reads CA registration sources. Implemented sources are defined in CertConfigLocation enumeration. Here is a quick example in my demo lab:

PS C:\> $CertConfig = New-Object SysadminsLV.PKI.Dcom.Implementations.CertConfigD
PS C:\> $CertConfig.EnumConfigEntries()


ComputerName         : dc1.contoso.com
CommonName           : Contoso CA
DisplayName          : Contoso CA
Description          :
OrganizationUnit     :
Organization         :
StateProvince        :
Locality             :
Country              :
ConfigString         : dc1.contoso.com\Contoso CA
Flags                : DsEntry, Registry, RegistryParent
SanitizedName        : Contoso CA
ShortName            : Contoso CA
SanitizedShortName   : Contoso CA
WebEnrollmentServers : {}

ComputerName         : dc2.contoso.com
CommonName           : contoso-DC2-CA
<...>


PS C:\> $CertConfig.FindConfigEntryByServerName("stdca01.contoso.com")


ComputerName         : StdCA01.contoso.com
CommonName           : Contoso Standalone CA01
DisplayName          : Contoso Standalone CA01
Description          :
OrganizationUnit     : PKI
Organization         : Contoso Pharmaceuticals
StateProvince        :
Locality             : Riga
Country              : LV
ConfigString         : StdCA01.contoso.com\Contoso Standalone CA01
Flags                : DsEntry
SanitizedName        : Contoso Standalone CA01
ShortName            : Contoso Standalone CA01
SanitizedShortName   : Contoso Standalone CA01
WebEnrollmentServers : {}



PS C:\>

First command retrieves all registered CAs. Alternatively, we can retrieve specific registration by either, CA host or certificate name. In last call I use filtering by host name. This interface is actively used in CertificateAuthority class.

Certification Authority property values

Another very big improvement is wrapping ICertRequest2::GetCAProperty and ICertAdmin2::GetCAProperty methods. This method isn’t very user-friendly, it has a lot of dependencies between parameter values, documentation is unclear and vague. After all, return value for this method is System.Object, so callers must know return type in advance and carefully use explicit cast. I’ve extracted this single method and refactored into separate class with a number of methods that correspond to property names and simplified the usage with strongly-typed return values (instead of generic System.Object). Abstract interface is ICertPropReaderD and implementation in CertPropReaderD class. Class constructor accepts CA configuration string and optional value whether ICertAdmin or ICertRequest underlying COM interface is used. The only difference between two is that ICertAdmin2 requires CA administrator permissions to activate, ICertRequest doesn’t. Constructor defaults to ICertRequest, so no need to have CA admin permissions to call methods in this class.

Certification Authority configuration values

Microsoft ADCS API supports CA configuration management (read and write values) using ICertAdmin2::GetConfigEntry method. When I explored this API, I was unable to access several configuration entries and moved to remote registry functionality as primary method and ICertAdmin::GetConfigEntry as a backup which wasn’t stable enough. After working closely with Microsoft Docs team I was able to match functionality provided by remote registry and this API. As the result, we have two methods to access CA configuration settings:

  • Remote Registry which relies on remote registry and permissions on remote CA server. Can read/write configuration when CA service is stopped. This transport is implemented in CertSrvRegManager class.
  • DCOM which relies on DCOM transport (do not require remote registry). Can read/write configuration only when CA service is running. This transport is implemented in CertSrvRegManagerD class.

Both classes implement same ICertRegManagerD interface and provide same functionality implemented in different ways. Here are features of this interface:

In addition, there is a combined class CertSrvConfigUtil which uses both implementations in fall-back mode. When invoking methods, the class checks if remote registry is accessible. If it is not, class attempts to use DCOM transport. Currently, this composite class supports only reading operations. I will add modification operations in next version. Writing operations are still supported in individual implementations: CertSrvRegManager and CertSrvRegManagerD. Let’s take a look at how it works.
Microsoft CA configuration is stored in registry under
HKLM:\SYSTEM\CurrentControlSet\Services\CertSvc\Configuration:

This node contains settings which are common to all instances of Microsoft CA. This is so-called Root Configuration Node. Root node contains Active string property that points to an active CA instance. That is, you may have multiple CA instances on single host and only one instance can be active and running at any time. Every CA instance is represented as subnode under root configuration node and is called an Active Configuration Node:

I’ve translated these two nodes to contexts: root context and active context:

PS C:\> $ConfigReader = New-Object SysadminsLV.PKI.Dcom.Implementations.CertSrvRegManagerD "dc2"
PS C:\> $ConfigReader

ComputerName                                                       IsAccessible ActiveConfig
------------                                                       ------------ ------------
dc2                                                                        True contoso-DC2-CA


PS C:\> $ConfigReader.GetStringConfigEntry("DBDirectory")
C:\Windows\system32\CertLog
PS C:\> $ConfigReader.GetNumericConfigEntry("SetupStatus")
24579
PS C:\>

By default, we refer to root context. Switching between contexts is done by using ICertRegManagerD.SetRootNode method. When forceActive method parameter is True, we switch to active context and to root context otherwise:

PS C:\> $ConfigReader.SetRootNode($true)
PS C:\> $ConfigReader.GetMultiStringConfigEntry("CACertPublicationURLs")
1:C:\Windows\System32\certsrv\CertEnroll\%1_%3%4.crt
0:http://%1/CertEnroll/%1_%3%4.crt
1:file://gw.contoso.com/pki/dc2ica%4.crt
2:http://www.contoso.com/pki/dc2ica%4.crt
32:http://dc2.contoso.com/ocsp
PS C:\> $ConfigReader.GetStringConfigEntry("CRLPeriod")
Weeks
PS C:\> $ConfigReader.GetNumericConfigEntry("CRLPeriodUnits")
1
PS C:\> $ConfigReader.GetBooleanConfigEntry("UseDS")
True
PS C:\>

As we can see in registry tree, there are subnodes under CA node, how do we access them? Easily, every read/write method in ICertRegManagerD interface has optional node parameter. It is set to null if you deal with entries directly under active CA node. In order to access CSP node, for example, you specify subnode or subnode path:

PS C:\> $ConfigReader.GetConfigEntry("Provider","CSP")
Microsoft Software Key Storage Provider
PS C:\> $ConfigReader.GetConfigEntry("CNGHashAlgorithm","CSP")
SHA256
PS C:\> $ConfigReader.GetConfigEntry("EventFilter","ExitModules\CertificateAuthority_MicrosoftDefault.Exit\SMTP")
8
PS C:\>

Writing value is as simple as reading:

PS C:\> $ConfigReader.SetConfigEntry

OverloadDefinitions
-------------------
void SetConfigEntry(System.Object data, string entryName, string node)
void ICertRegManagerD.SetConfigEntry(System.Object data, string entryName, string node)



PS C:\> $ConfigReader.SetConfigEntry(5,"CRLPeriod")
PS C:\> $ConfigReader.SetConfigEntry("Days","CRLPeriodUnits")
PS C:\> $ConfigReader.GetConfigEntry("CRLPeriod")
5
PS C:\> $ConfigReader.GetConfigEntry("CRLPeriodUnits")
Days
PS C:\>

Do not forget that CA service must be restarted to apply changes.

In next post, I will show more features we’ve added in PSPKI module and APIs.

Related Resources

  • Blog
    July 6, 2021

    Register TLS certificate with Remote Desktop Service using PowerShell

    Certificates, PowerShell, RDP
  • Blog
    May 7, 2021

    Just Released – Licensing Options for Our PKI Tools

    Development, PKI, PowerShell, Products, PSFCIV, PSPKI
  • Blog
    March 29, 2021

    PowerShell File Checksum Integrity Verifier (PsFCIV)

    PowerShell, PSFCIV

Vadims Podāns

PKI Software Architect

View All Posts by Vadims Podāns

Comments

  • Yeah I have the whole submit request, retrieve certificate, install certificate process working.

    To be able to use the client certificate for iisClientCertificateMappings I have to export the certificate from the user’s Userstore either manually or programmatically which works fine.

    Is it possible to avoid this step and convert the response from the CA into the same base64 string that is exported from the UserStore?

    If I just convert the CA certificate response supplied by GetIssuedCertificate to base64 it doesn’t work.

  • I’ve currently hit a wall with netcore project where I think i need to use certadm.dll on windows 10 but I can only add a com reference for certenroll and certclilib.

    Im trying to export a base 64 version of a client cert directly from the CA so the output can be used for IIS client certificate mappings.

    If I export from the user store I get a working base 64 cert that works with IIS client cert auth mappings.

    I struggled with just converting the CA response for an issued cert to base 64 but obviously that isn’t going to cut it.

    Will GetCAProperty lead me down the right path to getting the BinaryCertificate value as if I was manually exporting it from the admin UI?

    All examples I find on the web seem to require certadm.dll and this isn’t available on my win 10 machine even with the windows platform sdk installed.

    • > Will GetCAProperty lead me down the right path to getting the BinaryCertificate
      no, GetCAProperty retrieves only CA configuration property, it doesn’t retrieve anything from CA database.

      You can get issued certificate using ICertRequest2::GetIssuedCertificate if you know request ID in advance.

Leave a Reply

Your email address will not be published. Required fields are marked *